1. 程式人生 > >ionic+cordova基於websocket實現的實時通報提醒功能

ionic+cordova基於websocket實現的實時通報提醒功能

app接收後臺發出的請求,並通過狀態列提示使用者有新的訊息,這裡首先要解決的就是前後端如何互聯,像pc端的應用,一般會採用前端定時請求後臺,但如果要app定時去訪問後臺的話,對使用者來說並不友好,這會消耗使用者大量的移動流量,移動端最好的方式就是後臺主動向app推送資訊,h5提供了一種比較好的方式就是websocket,使用者開啟app後,向後臺發出請求,後臺相應後,就可以實時向前端推送資訊了,而無需app再次去訪問,具體前後臺實現如下:

package com.wavefax.wavefax51.openapi.servlet;

import java.io.IOException;
import java.util.concurrent.CopyOnWriteArraySet;
 
import javax.websocket.OnClose;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.ServerEndpoint;

/**
 *
 * @author Administrator
 */
@ServerEndpoint("/websocket")
public class WebSocket {
    //靜態變數,用來記錄當前線上連線數。應該把它設計成執行緒安全的。
    private static int onlineCount = 0;
     
    //concurrent包的執行緒安全Set,用來存放每個客戶端對應的MyWebSocket物件。若要實現服務端與單一客戶端通訊的話,可以使用Map來存放,其中Key可以為使用者標識
    private static CopyOnWriteArraySet<WebSocket> webSocketSet = new CopyOnWriteArraySet<WebSocket>();
     
    //與某個客戶端的連線會話,需要通過它來給客戶端傳送資料
    private Session session;
     
    /**
     * 連線建立成功呼叫的方法
     * @param session  可選的引數。session為與某個客戶端的連線會話,需要通過它來給客戶端傳送資料
     */
    @OnOpen
    public void onOpen(Session session){
        this.session = session;
        webSocketSet.add(this);     //加入set中
        addOnlineCount();           //線上數加1
        System.out.println("有新連線加入!當前線上人數為" + getOnlineCount());
    }
     
    /**
     * 連線關閉呼叫的方法
     */
    @OnClose
    public void onClose(){
        webSocketSet.remove(this);  //從set中刪除
        subOnlineCount();           //線上數減1    
        System.out.println("有一連線關閉!當前線上人數為" + getOnlineCount());
    }
     
    /**
     * 收到客戶端訊息後呼叫的方法
     * @param message 客戶端傳送過來的訊息
     * @param session 可選的引數
     */
    @OnMessage
    public void onMessage(String message, Session session) {
        System.out.println("來自客戶端的訊息:" + message);
         
        //群發訊息
        for(WebSocket item: webSocketSet){             
            try {
                item.sendMessage(message);
            } catch (IOException e) {
                e.printStackTrace();
                continue;
            }
        }
    }
     
    /**
     * 發生錯誤時呼叫
     * @param session
     * @param error
     */
    @OnError
    public void onError(Session session, Throwable error){
        System.out.println("發生錯誤");
        error.printStackTrace();
    }
     
    /**
     * 這個方法與上面幾個方法不一樣。沒有用註解,是根據自己需要新增的方法。
     * @param message
     * @throws IOException
     */
    public void sendMessage(String message) throws IOException{
        this.session.getBasicRemote().sendText(message);
        //this.session.getAsyncRemote().sendText(message);
    }
 
    public static synchronized int getOnlineCount() {
        return onlineCount;
    }
 
    public static synchronized void addOnlineCount() {
        WebSocket.onlineCount++;
    }
     
    public static synchronized void subOnlineCount() {
        WebSocket.onlineCount--;
    }    
}

在app發請求的程式碼如下:

websocket = null;

var websocketAddress = 'ws://'+ url

//判斷當前瀏覽器是否支援WebSocket
if('WebSocket' in window){
    websocket = new WebSocket(websocketAddress);
}
else{
    alert('Not support websocket')
}

//連線發生錯誤的回撥方法
websocket.onerror = function(){
    //notificationReminder("error");
};

//連線成功建立的回撥方法
websocket.onopen = function(event){
    console.log(event);
}

//接收到訊息的回撥方法
websocket.onmessage = function(event){
    $scope.notificationReminder(event.data);
}

//連線關閉的回撥方法
websocket.onclose = function(){
    //notificationReminder("close");
}

//監聽視窗關閉事件,當視窗關閉時,主動去關閉websocket連線,防止連線還沒斷開就關閉視窗,server端會拋異常。
window.onbeforeunload = function(){
    websocket.close();
}

//傳送訊息
$scope.send = function(){
    websocket.send(localStorageService.get('UserID'));
}
$scope.closeWebSocket = function(){
    websocket.close();
}


前後端互聯成功後,要做就是前端接收到後臺發來的資訊後,如何顯示在狀態列上,提示使用者,這裡主要用到cordova提供的外掛,首先安裝cordova-plugin-local-notifications,然後在js配置好如下就可以了:

狀態列通報提醒
$scope.notificationReminder = function(faxInfo){
    cordova.plugins.notification.local.add({  
        id: id,
        title: '',  
        text: '',  
        at: new Date().getTime(),  
        //badge: 2,  
        autoClear: true,//預設值  
        sound: 'res://platform_default',//預設值  
        icon: 'res://ic_popup_reminder', //預設值  
        ongoing: false  //預設值  
    });
}