好湿?好紧?好多水好爽自慰,久久久噜久噜久久综合,成人做爰A片免费看黄冈,机机对机机30分钟无遮挡

主頁 > 知識庫 > golang websocket 服務端的實現

golang websocket 服務端的實現

熱門標簽:學海導航地圖標注 南通如皋申請開通400電話 高德地圖標注口訣 地圖標注的汽車標 中國地圖標注省會高清 浙江高速公路地圖標注 廣州呼叫中心外呼系統 江西轉化率高的羿智云外呼系統 西部云谷一期地圖標注

創建一個websocket的服務端

package smile

import (
  "errors"
  "log"
  "net/http"
  "sync"
  "time"

  "github.com/gorilla/websocket"
)

const (
  // 允許等待的寫入時間
  writeWait = 10 * time.Second

  // Time allowed to read the next pong message from the peer.
  pongWait = 60 * time.Second

  // Send pings to peer with this period. Must be less than pongWait.
  pingPeriod = (pongWait * 9) / 10

  // Maximum message size allowed from peer.
  maxMessageSize = 512
)

// 最大的連接ID,每次連接都加1 處理
var maxConnId int64

// 客戶端讀寫消息
type wsMessage struct {
  // websocket.TextMessage 消息類型
  messageType int
  data    []byte
}

// ws 的所有連接
// 用于廣播
var wsConnAll map[int64]*wsConnection

var upgrader = websocket.Upgrader{
  ReadBufferSize: 1024,
  WriteBufferSize: 1024,
  // 允許所有的CORS 跨域請求,正式環境可以關閉
  CheckOrigin: func(r *http.Request) bool {
    return true
  },
}

// 客戶端連接
type wsConnection struct {
  wsSocket *websocket.Conn // 底層websocket
  inChan  chan *wsMessage // 讀隊列
  outChan chan *wsMessage // 寫隊列

  mutex   sync.Mutex // 避免重復關閉管道,加鎖處理
  isClosed bool
  closeChan chan byte // 關閉通知
  id    int64
}

func wsHandler(resp http.ResponseWriter, req *http.Request) {
  // 應答客戶端告知升級連接為websocket
  wsSocket, err := upgrader.Upgrade(resp, req, nil)
  if err != nil {
    log.Println("升級為websocket失敗", err.Error())
    return
  }
  maxConnId++
  // TODO 如果要控制連接數可以計算,wsConnAll長度
  // 連接數保持一定數量,超過的部分不提供服務
  wsConn := wsConnection{
    wsSocket: wsSocket,
    inChan:  make(chan *wsMessage, 1000),
    outChan:  make(chan *wsMessage, 1000),
    closeChan: make(chan byte),
    isClosed: false,
    id:    maxConnId,
  }
  wsConnAll[maxConnId] = wsConn
  log.Println("當前在線人數", len(wsConnAll))

  // 處理器,發送定時信息,避免意外關閉
  go wsConn.processLoop()
  // 讀協程
  go wsConn.wsReadLoop()
  // 寫協程
  go wsConn.wsWriteLoop()
}

// 處理隊列中的消息
func (wsConn *wsConnection) processLoop() {
  // 處理消息隊列中的消息
  // 獲取到消息隊列中的消息,處理完成后,發送消息給客戶端
  for {
    msg, err := wsConn.wsRead()
    if err != nil {
      log.Println("獲取消息出現錯誤", err.Error())
      break
    }
    log.Println("接收到消息", string(msg.data))
    // 修改以下內容把客戶端傳遞的消息傳遞給處理程序
    err = wsConn.wsWrite(msg.messageType, msg.data)
    if err != nil {
      log.Println("發送消息給客戶端出現錯誤", err.Error())
      break
    }
  }
}

// 處理消息隊列中的消息
func (wsConn *wsConnection) wsReadLoop() {
  // 設置消息的最大長度
  wsConn.wsSocket.SetReadLimit(maxMessageSize)
  wsConn.wsSocket.SetReadDeadline(time.Now().Add(pongWait))
  for {
    // 讀一個message
    msgType, data, err := wsConn.wsSocket.ReadMessage()
    if err != nil {
      websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway, websocket.CloseAbnormalClosure)
      log.Println("消息讀取出現錯誤", err.Error())
      wsConn.close()
      return
    }
    req := wsMessage{
      msgType,
      data,
    }
    // 放入請求隊列,消息入棧
    select {
    case wsConn.inChan - req:
    case -wsConn.closeChan:
      return
    }
  }
}

// 發送消息給客戶端
func (wsConn *wsConnection) wsWriteLoop() {
  ticker := time.NewTicker(pingPeriod)
  defer func() {
    ticker.Stop()
  }()
  for {
    select {
    // 取一個應答
    case msg := -wsConn.outChan:
      // 寫給websocket
      if err := wsConn.wsSocket.WriteMessage(msg.messageType, msg.data); err != nil {
        log.Println("發送消息給客戶端發生錯誤", err.Error())
        // 切斷服務
        wsConn.close()
        return
      }
    case -wsConn.closeChan:
      // 獲取到關閉通知
      return
    case -ticker.C:
      // 出現超時情況
      wsConn.wsSocket.SetWriteDeadline(time.Now().Add(writeWait))
      if err := wsConn.wsSocket.WriteMessage(websocket.PingMessage, nil); err != nil {
        return
      }
    }
  }
}

// 寫入消息到隊列中
func (wsConn *wsConnection) wsWrite(messageType int, data []byte) error {
  select {
  case wsConn.outChan - wsMessage{messageType, data}:
  case -wsConn.closeChan:
    return errors.New("連接已經關閉")
  }
  return nil
}

// 讀取消息隊列中的消息
func (wsConn *wsConnection) wsRead() (*wsMessage, error) {
  select {
  case msg := -wsConn.inChan:
    // 獲取到消息隊列中的消息
    return msg, nil
  case -wsConn.closeChan:

  }
  return nil, errors.New("連接已經關閉")
}

// 關閉連接
func (wsConn *wsConnection) close() {
  log.Println("關閉連接被調用了")
  wsConn.wsSocket.Close()
  wsConn.mutex.Lock()
  defer wsConn.mutex.Unlock()
  if wsConn.isClosed == false {
    wsConn.isClosed = true
    // 刪除這個連接的變量
    delete(wsConnAll, wsConn.id)
    close(wsConn.closeChan)
  }
}

// 啟動程序
func StartWebsocket(addrPort string) {
  wsConnAll = make(map[int64]*wsConnection)
  http.HandleFunc("/ws", wsHandler)
  http.ListenAndServe(addrPort, nil)
}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • golang基于websocket實現的簡易聊天室程序
  • Golang使用WebSocket通信的實現
  • 一百行Golang代碼實現簡單并發聊天室
  • golang實現一個簡單的websocket聊天室功能

標簽:貴州 常州 保定 德宏 曲靖 吐魯番 許昌 東營

巨人網絡通訊聲明:本文標題《golang websocket 服務端的實現》,本文關鍵詞  golang,websocket,服務,端,的,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《golang websocket 服務端的實現》相關的同類信息!
  • 本頁收集關于golang websocket 服務端的實現的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 免费看女人隐私秘?部位视频洗澡| 娇小初叫videos摘花第一次| 久久r热这里有精品视频| 一男多女h| 成品视频大全观视频的技巧有哪些| 国产清高无码色情观看| 香蕉在线精品亚洲第一区| 中国美女撒尿TXXXX视频偷窥| 欧美人体白嫩极品hotpics| 精品欧美AV无码喷奶水| 免费网站无码秘?白丝视频东京熱| 一区二区国产在线播放| 男男被啪得大叫gif动态图| 久久精品国产久金国产思思| 一级做a爱过程免费视频高清| 人妻人人做人做人人爱双色球| 国产精品久久久久久无码公交车站| 韩国激情禁片健身教练| 英语老师今晚都是你的了| 国产精品嫩草研究院成人| 秘密教学84这次换我教你了| 欧美精品免费看| 男女啪啪做爰高潮全过有多长| 男人天堂国产| 国产爽的冒白浆的视频高清| 亚洲国产国产综合一区首页| 双乳被两个男人揉着玩弄的 | 午夜理伦大片一级| 亚洲美女精品| 免费无码婬片A片AAA毛在线看| 免费看欧美成人A片无码| 女上男下xx00xx00视频网站| 古风一女n夫到处做高h| 东北浓毛女HD脏话对白熟女| 老师扒开腿秘?让我爽了外国| ??爱情岛亚洲首页论坛 | 仙子被强行双修巨大进入后面 | 寡妇高潮一级毛片免费观看A片| 大手伸进她的亵衣h| 美国户外撒尿xxxx| 羞羞网页登界面入口|