nodejs 開發(fā)websocket 筆記

使用nodejs 開發(fā)websocket消息分發(fā)系統(tǒng)

websocket使用事件觸發(fā)機(jī)制傳輸密集傳輸數(shù)據(jù),跟nodejs的事件機(jī)制相同,比起早期comet技術(shù)使用的http響應(yīng)式更為高效,客戶端不用重復(fù)的與服務(wù)端握手更加節(jié)約服務(wù)器性能。

1、在客戶端調(diào)用

socket事件

message 消息接收
open 與服務(wù)器握手完成后觸發(fā)相對(duì)與服務(wù)端的upgrade事件
close socket關(guān)閉
error 錯(cuò)誤消息
ws.send 發(fā)送消息,消息內(nèi)容必須為字符串或buffer

let url = "ws://127.0.0.1:3000/socket?id="+id+"&name="+name+"&gid="+teamid;

let ws= new WebSocket(url);

ws.addEventListener("message",this.message.bind(this));
ws.addEventListener("open",this.open.bind(this));
ws.addEventListener("close",this.close.bind(this));
ws.addEventListener("error",this.error.bind(this));

ws.send("消息發(fā)送");
2、服務(wù)端接收調(diào)用

在服務(wù)端使用koa-websocket

其響應(yīng)原理為客戶端請(qǐng)求后,服務(wù)端切換響應(yīng)為websocket,在upgrade事件中

1、收到客戶端請(qǐng)求頭

upgrade:websocket
connection:upgrade
Sec-WebSocket-Key:xxxx==

2、服務(wù)端使用258EAFA5-E914-47DA-95CA-C5AB0DC85B11通過sha1散列算法合并key手動(dòng)修改Sec-WebSocket-Key 返回客戶端完成
3、upgrade事件參數(shù)req,socket,upgradehead,

socket.setNoDelay(true)
socket.write(修改過端返回頭)
websocket = new WebSocfket()
websocket.setSocket(socket) 創(chuàng)建websocket鏈接

message響應(yīng)客戶端事件 ws.send() 發(fā)送客戶端事件

let ws = ctx.websocket;
    ws.on("message",this.EVNET_MESSAGE.bind(this));
    ws.on("error",this.EVENT_ERROR.bind(this));
    ws.on("close",this.EVENT_CLOSE.bind(this));
    ws.on("timeout",this.EVENT_ERROR.bind(this));  
 

3、在后期開發(fā)中加入了redis與mssql 來進(jìn)行數(shù)據(jù)存儲(chǔ)操作

因?yàn)閞edis自己有重連機(jī)制,不需要在處理重新連接,與sql數(shù)據(jù)庫操作不同,不用重復(fù)端去調(diào)用初始化,此處為坑

 this.client = redis.createClient({
            host:host,port:port,
            retry_strategy(options){
                if (options.error.code === 'ECONNREFUSED') {
                    
                    console.log('連接被拒絕');
                    Log.error('連接被拒絕');
                }
                if (options.times_connected > 10) {
                    console.log('重試連接超過十次');
                    Log.error('重試連接超過十次');  
                    
                }
                return Math.max(options.attempt * 100, 3000);
            }
        });

4、mssql

使用 new sql.ConnectionPool(this.config).connect();創(chuàng)建鏈接
在查詢完成后sql.close();來關(guān)閉鏈接,跟api上說明使用pool.close()不同,pool會(huì)造成二次連接數(shù)據(jù)庫失敗。此處為坑


const sql = require('mssql');
class xx{

connect(){  //創(chuàng)建鏈接
        return new sql.ConnectionPool(this.config).connect();
        
    }
    
    async query(str){  //查詢數(shù)據(jù)
        console.log("查詢或入庫")
        let pool;
        let result;
     try{
        pool = await  this.connect()
        result = await pool.request().query(str)
        }catch(e){
            console.log("數(shù)據(jù)庫鏈接錯(cuò)誤:"+e.message);
            Log.error("數(shù)據(jù)庫鏈接錯(cuò)誤:"+e.message);
            return Promise.reject(e.message).catch(str=>{
                this.emit("DBError",str);
            })
        }
      
       
      let rows = result.recordset; 
      let data=[]
         if(result.recordsets.length){
            data = result.recordsets[0];
            }     
           sql.close();
         return Promise.resolve(toJson(data))    
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容