成人性生交大片免费看视频r_亚洲综合极品香蕉久久网_在线视频免费观看一区_亚洲精品亚洲人成人网在线播放_国产精品毛片av_久久久久国产精品www_亚洲国产一区二区三区在线播_日韩一区二区三区四区区区_亚洲精品国产无套在线观_国产免费www

主頁 > 知識庫 > 使用Go基于WebSocket構(gòu)建千萬級視頻直播彈幕系統(tǒng)的代碼詳解

使用Go基于WebSocket構(gòu)建千萬級視頻直播彈幕系統(tǒng)的代碼詳解

熱門標(biāo)簽:濮陽自動外呼系統(tǒng)代理 賺地圖標(biāo)注的錢犯法嗎 福州鐵通自動外呼系統(tǒng) 烏魯木齊人工電銷機(jī)器人系統(tǒng) 廣東語音外呼系統(tǒng)供應(yīng)商 智能電銷機(jī)器人營銷 地圖標(biāo)注測試 澳門防封電銷卡 長沙ai機(jī)器人電銷

(1)業(yè)務(wù)復(fù)雜度介紹

開門見山,假設(shè)一個直播間同時500W人在線,那么1秒鐘1000條彈幕,那么彈幕系統(tǒng)的推送頻率就是: 500W * 1000條/秒=50億條/秒 ,想想B站2019跨年晚會那次彈幕系統(tǒng)得是多么的NB,況且一個大型網(wǎng)站不可能只有一個直播間!

使用Go做WebSocket開發(fā)無非就是三種情況:

  • 使用Go原生自帶的庫,也就是 golang.org/x/net ,但是這個官方庫真是出了奇Bug多
  • 使用GitHub大佬 gorilla/websocket 庫,可以結(jié)合到某些Web開發(fā)框架,比如Gin、iris等,只要使用的框架式基于 golang.org/net 的,那么這個庫就可以與這個框架結(jié)合
  • 手?jǐn)]一個WebSocket框架

根據(jù)估算結(jié)果,彈幕推送量很大的時候,Linux內(nèi)核將會出現(xiàn)瓶頸,因為Linux內(nèi)核發(fā)送TCP包的時候極限包發(fā)送頻率是100W。因此可以將同一秒內(nèi)的彈幕消息合并為1條推送,減少網(wǎng)絡(luò)小數(shù)據(jù)包的發(fā)送,從而降低推送頻率。

彈幕系統(tǒng)需要維護(hù)在線的用戶長連接來實現(xiàn)定向推送到在線的用戶,通常是使用Hash字典結(jié)構(gòu),通常推送消息就是遍歷在線用的Hash字典。在彈幕推送期間用戶在不斷的上下線,為了維護(hù)上線用戶,那么就得不斷的修改Hash字典,不斷地進(jìn)行鎖操作,用戶量過大導(dǎo)致鎖瓶頸。因此可以將整個Hash結(jié)構(gòu)拆分為多個Hash結(jié)構(gòu),分別對多個Hash結(jié)構(gòu)加不同的鎖,并且使用讀寫鎖替代互斥鎖。

通常服務(wù)器與客戶端交互使用JSON結(jié)構(gòu),那么需要不斷的編碼解碼JSON數(shù)據(jù),這將會導(dǎo)致CPU瓶頸。將消息先進(jìn)行合并,然后進(jìn)行編碼,最后輪詢Hash結(jié)構(gòu)進(jìn)行推送。

以上是單體架構(gòu)存在的問題,為了支持更多的用戶負(fù)載,通常彈幕系統(tǒng)采用分布式架構(gòu),進(jìn)行彈性擴(kuò)容縮容。

(2)推送還是拉取?

如果是客戶端拉取服務(wù)器端數(shù)據(jù),那么將會存在以下幾個問題:

  • 直播在線人數(shù)多就意味著消息數(shù)據(jù)更新頻率高,拉取消息意味著彈幕無法滿足時效性
  • 如果很多客戶端同時拉取,那么服務(wù)器端的壓力無異于DDOS
  • 一個彈幕系統(tǒng)應(yīng)該是通用的,因此對于直播間彈幕較少的場景,意味著消息數(shù)據(jù)拉取請求都是無效的

因此我們考慮推送模式:當(dāng)數(shù)據(jù)發(fā)生更新的時候服務(wù)器端主動推送到客戶端,這樣可以有效減少客戶端的請求次數(shù)。如果需要實現(xiàn)消息推送,那么就意味著服務(wù)器端維護(hù)大量的長連接。

(3)為什么使用WebSocket?

實現(xiàn)彈幕消息的實時更新一定是使用Socket的方式,那么為啥要使用WebSocket呢?現(xiàn)在大部分直播應(yīng)用的開發(fā)都是跨平臺的,然而跨平臺的開發(fā)框架本質(zhì)就是Web開發(fā),那么一定離不開WebSocket,而且一部分用戶會選擇在Web端看視頻,比如Bilibili,現(xiàn)如今也有一些桌面應(yīng)用是用Electron等跨平臺框架開發(fā)的,比如Lark飛書等,因此實現(xiàn)消息推送的最佳方案就是使用WebSocket。

使用WebSocket可以輕松的維持服務(wù)器端長連接,其次WebSocket是架構(gòu)在HTTP協(xié)議之上的,并且也可以使用HTTPS方式,因此WebSocket是可靠傳輸,并且不需要開發(fā)者關(guān)注底層細(xì)節(jié)。

為啥要使用Go搞WebSocket呢?首先說到WebSocket你可能會想到Node.js,但是Node.js是單線程模型,如果實現(xiàn)高并發(fā),不得不創(chuàng)建多個Node.js進(jìn)程,但是這又不容易服務(wù)端遍歷整個連接集合;如果使用Java就會顯得比較笨重,Java項目的部署,編寫Dockerfile都不如Go的目標(biāo)二進(jìn)制更加簡潔,并且Go協(xié)程很容易實現(xiàn)高并發(fā),上一章說到Go語言目前也有成熟的WebSocket輪子。

(4)服務(wù)端基本Demo

首先搭建好一個框架:

package main

import (
  "fmt"
  "net/http"
)

func main() {
 fmt.Println("Listen localhost:8080")
   // 注冊一個用于WebSocket的路由,實際業(yè)務(wù)中不可能只有一個路由
  http.HandleFunc("/messages", messageHandler)
  // 監(jiān)聽8080端口,沒有實現(xiàn)服務(wù)異常處理器,因此第二個參數(shù)是nil
  http.ListenAndServe("localhost:8080", nil)
}

func messageHandler(response http.ResponseWriter, request *http.Request) {
  // TODO: 實現(xiàn)消息處理
  response.Write([]byte("HelloWorld"))
}

然后完善messageHandler函數(shù):

func messageHandler(response http.ResponseWriter, request *http.Request) {
  var upgrader = websocket.Upgrader{
    // 允許跨域
    CheckOrigin: func(resquest *http.Request) bool {
      return true
    },
  }

  // 建立連接
  conn, err := upgrader.Upgrade(response, request, nil)
  if err != nil {
    return
  }

  // 收發(fā)消息
  for {
    // 讀取消息
    _, bytes, err := conn.ReadMessage()
    if err != nil {
      _ = conn.Close()
    }
    // 寫入消息
    err = conn.WriteMessage(websocket.TextMessage, bytes)
    if err != nil {
      _ = conn.Close()
    }
  }
}

現(xiàn)在基本上實現(xiàn)了WebSocket功能,但是websocket的原生API不是線程安全的(Close方法是線程安全的,并且是可重入的),并且其他模塊無法復(fù)用業(yè)務(wù)邏輯,因此進(jìn)行封裝:

  • 封裝Connection對象描述一個WebSocket連接
  • 為Connection對象提供線程安全的關(guān)閉、接收、發(fā)送API
// main.go
package main

import (
  "bluemiaomiao.github.io/websocket-go/service"
  "fmt"
  "net/http"

  "github.com/gorilla/websocket"
)

func main() {
  fmt.Println("Listen localhost:8080")
  http.HandleFunc("/messages", messageHandler)
  _ = http.ListenAndServe("localhost:8080", nil)
}

func messageHandler(response http.ResponseWriter, request *http.Request) {
  var upgrader = websocket.Upgrader{
    // 允許跨域
    CheckOrigin: func(resquest *http.Request) bool {
      return true
    },
  }

  // 建立連接
  conn, err := upgrader.Upgrade(response, request, nil)
  wsConn, err := service.Create(conn)
  if err != nil {
    return
  }

  // 收發(fā)消息
  for {
    // 讀取消息
    msg, err := wsConn.ReadOne()
    if err != nil {
      wsConn.Close()
    }
    // 寫入消息
    err = wsConn.WriteOne(msg)
    if err != nil {
      _ = conn.Close()
    }
  }
}
// service/messsage_service.go
package service

import (
  "errors"
  "github.com/gorilla/websocket"
  "sync"
)

// 封裝的連接對象
// 
// 由于websocket的Close()方法是可重入的,所以可以多次調(diào)用,但是關(guān)閉Channel的close()
// 方法不是可重入的,因此通過isClosed進(jìn)行判斷
// isClosed可能發(fā)生資源競爭,因此通過互斥鎖避免
// 關(guān)閉websocket連接后,也要自動關(guān)閉輸入輸出消息流,因此通過signalCloseLoopChan實現(xiàn)
type Connection struct {
  conn                  *websocket.Conn  // 具體的連接對象
  inputStream             chan []byte       // 輸入流,使用Channel模擬
  outputStream           chan []byte       // 輸出流,使用chaneel模擬
  signalCloseLoopChan     chan byte       // 關(guān)閉信號
  isClosed               bool            // 是否調(diào)用過close()方法
  lock                   sync.Mutex      // 簡單的鎖
}

// 用于初始化一個連接對象
func Create(conn *websocket.Conn) (connection *Connection, err error) {
  connection = Connection{
    conn:              conn,
    inputStream:        make(chan []byte, 1000),
    outputStream:       make(chan []byte, 1000),
    signalCloseLoopChan: make(chan byte, 1),
    isClosed:            false,
  }

  // 啟動讀寫循環(huán)
  go connection.readLoop()
  go connection.writeLoop()
  return
}

// 讀取一條消息
func (c *Connection) ReadOne() (msg []byte, err error) {
  select {
  case msg = -(*c).inputStream:
  case -(*c).signalCloseLoopChan:
    err = errors.New("connection is closed")
  }
  return
}

// 寫入一條消息
func (c *Connection) WriteOne(msg []byte) (err error) {
  select {
  case (*c).outputStream - msg:
  case -(*c).signalCloseLoopChan:
    err = errors.New("connection is closed")
  }
  return
}

// 關(guān)閉連接對象
func (c *Connection) Close() {
  _ = (*c).conn.Close()
  (*c).lock.Lock()
  if !(*c).isClosed {
    close((*c).signalCloseLoopChan)
  }
  (*c).lock.Unlock()

}

// 讀取循環(huán)
func (c *Connection) readLoop() {
  // 不停的讀取長連接中的消息,只要存在消息就將其放到隊列中
  for {
    _, bytes, err := (*c).conn.ReadMessage()
    if err != nil {
      (*c).Close()
    }
    select {
    case -(*c).signalCloseLoopChan:
      (*c).Close()
    case (*c).inputStream - bytes:
    }
  }
}

// 寫入循環(huán)
func (c *Connection) writeLoop() {
  // 只要隊列中存在消息,就將其寫入
  var data []byte
  for {
    select {
    case data = -(*c).outputStream:
    case -(*c).signalCloseLoopChan:
      (*c).Close()
    }
    err := (*c).conn.WriteMessage(websocket.TextMessage, data)
    if err != nil {
      _ = (*c).conn.Close()
    }
  }
}

至此,你已經(jīng)學(xué)會了如何使用Go構(gòu)建WebSocket服務(wù)。

到此這篇關(guān)于使用Go基于WebSocket構(gòu)建千萬級視頻直播彈幕系統(tǒng)的代碼詳解的文章就介紹到這了,更多相關(guān)go WebSocket視頻直播彈幕內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • golang 實現(xiàn)tcp server端和client端,并計算RTT時間操作
  • golang websocket 服務(wù)端的實現(xiàn)
  • golang socket斷點(diǎn)續(xù)傳大文件的實現(xiàn)方法
  • golang基于websocket實現(xiàn)的簡易聊天室程序
  • golang網(wǎng)絡(luò)socket粘包問題的解決方法
  • Golang 實現(xiàn)Socket服務(wù)端和客戶端使用TCP協(xié)議通訊

標(biāo)簽:西雙版納 阿克蘇 調(diào)研邀請 太原 貴陽 德州 廣西 慶陽

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《使用Go基于WebSocket構(gòu)建千萬級視頻直播彈幕系統(tǒng)的代碼詳解》,本文關(guān)鍵詞  使用,基于,WebSocket,構(gòu)建,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《使用Go基于WebSocket構(gòu)建千萬級視頻直播彈幕系統(tǒng)的代碼詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于使用Go基于WebSocket構(gòu)建千萬級視頻直播彈幕系統(tǒng)的代碼詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    欧美激情第一页在线观看| 老熟妇精品一区二区三区| 色综合一区二区三区| 成人18精品视频| 亚洲国产欧美日韩精品| 3344永久| 久久免费少妇高潮99精品| 欧美一区三区三区高中清蜜桃| 国语自产偷拍精品视频偷| 欧美激情中文字幕| 日韩视频在线观看一区二区三区| 欧美日韩精品一区视频| 一区在线播放| 午夜欧美理论片| 亚洲精品tv久久久久久久久久| av大片在线免费观看| 国产日韩av高清| 性一交一乱一伧老太| 日韩av影视综合网| 欧美性受xxxx白人性爽| 狠狠综合久久av一区二区老牛| 久久精品麻豆| 久久久久久久久久久久av| 成人免费视频网站在线看| 警花av一区二区三区| 日韩毛片免费视频一级特黄| 亚洲影院色无极综合| 日韩精品一二三四| 日批视频网站| 亚洲精品视频久久久| 日韩欧美aⅴ综合网站发布| 欧美日韩另类丝袜其他| 波多野结衣在线播放一区| 国产精品国内视频| 欧洲一级精品| 先锋av资源在线| 国产精品丝袜一区二区| 狠狠久久综合| 欧美在线aaa| 中文字幕人妻一区二区| eeuss鲁丝片eeuss影院| 日韩理论片av| 免费成人高清| 欧美在线一区二区三区四| 国产小视频在线高清播放| 国产成人精品一区| 国产夫妻视频| 美女欧美视频在线观看免费| 一区二区精品区| 欧美日韩一区在线观看| 无套内谢丰满少妇中文字幕| 在线天堂资源www在线污| 亚洲熟妇一区二区| 免费黄色在线视频网站| 亚洲av无码乱码在线观看性色| 99在线观看精品视频| metart日本精品嫩模| 日日夜夜精品网站| 国产69精品久久久久久久久久| 丁香六月天婷婷| 羞羞答答国产精品www一本| 天天综合精品| 国产美女直播视频一区| 国产一级中文字幕| 最新国产乱人伦偷精品免费网站| 日本不卡一区二区| 欧美少妇精品| 亚洲欧美视频一区| 中文在线观看免费网站| 成年人在线观看视频免费| 不要播放器的av网站| 国产原创中文av| 国产主播一区二区三区四区| 精品国产一区二区三区久久久久久| 91精品国产91久久久| 草草久久久无码国产专区| 色七七影院综合| 都市激情国产精品| 99久热re在线精品996热视频| 亚洲国产成人精品无码区99| 欧美午夜片在线免费观看| 日日夜夜精品网站| 极品一区美女高清| 成人91在线| 四虎成人在线观看| 欧美在线高清视频| 欧美日韩一区二区三区在线看| 116极品美女视频在线观看| 激情综合视频| 久久99久久99| 日本黄视频网站| 男人av在线播放| 日韩欧美三区| 一区二区三区中文字幕精品精品| 三级理论午夜在线观看| 动漫3d精品一区二区三区乱码| 国产调教打屁股xxxx网站| 日韩一级片在线免费观看| 偷拍一区二区| 在线观看免费p片视频网站地址| 久久久婷婷一区二区三区不卡| 在线观看亚洲欧美| 亚洲免费不卡| 看全色黄大色大片免费久久久| japanesexxxxfree在线观看| 国产精品久久久av| 亚洲色图欧美在线| 亚洲一区二区三区四区五区xx| 色视频免费在线观看| 99国产欧美久久久精品| 精品免费视频123区| 九九久久九九| 国产在线www| 99久久免费国| 欧美久久免费观看| 庆余年2免费日韩剧观看大牛| 一级片免费在线| 久久欧美一区二区| 日韩欧美中文视频| 又爽又大又黄a级毛片在线视频| 国内毛片毛片毛片毛片| 欧美久久精品一级黑人c片| 不卡视频一区二区三区| 亚洲乱码国产乱码精品精的特点| 中文字幕中文字幕在线中心一区| 久久精品国产精品亚洲| 爱爱视频网站免费| ,一级淫片a看免费| 成人亚洲欧美日韩在线观看| 日韩欧美在线观看强乱免费| 欧美高清视频在线播放| 亚洲精品永久www嫩草| 欧美精品羞羞答答| 又色又爽又黄18网站| 日中文字幕在线| 久久国产精品区| 91精品国产91久久久久久一区二区| 激情六月婷婷综合| 在线观看免费高清视频| 中文字幕综合网| 亚洲经典一区二区三区| 蜜桃视频污在线观看| 精品国产一区二区三区四区vr| 最近中文字幕在线中文视频| 亚洲精品视频二区| 亚洲精品一区二区三区香蕉| 亚洲欧美中文另类| 亚洲h在线观看| 国产又黄又猛又粗又爽| 99久久国产宗和精品1上映| 成人免费淫片95视频观看网站| 欧美成人激情图片网| 亚洲国产成人porn| 久久久婷婷一区二区三区不卡| 在线看的黄色网址| 久久久久成人片免费观看蜜芽| 国产真实老熟女无套内射| 午夜国产一区二区| 久久高清精品| 国产精品色呦| 大杳蕉精品视频在线观看| 欧美一区二区影院| 国产一区二区三区18| 精品人妻一区二区三区视频| 欧美一区二区三区在线电影| 国产日产欧美一区二区| 亚洲免费在线视频一区 二区| 乳奴隷乳フ辱julia在线观看| 免费一级a毛片夜夜看| 欧美在线观看一区二区三区| 国产精品www994| 国产网友自拍电影在线| 亚洲精品久久久成人| 成人午夜私人影院| 欧美mv日韩mv国产网站| 成人免费网站www网站高清| 国模人体一区二区| 性综艺节目av在线播放| 91深夜福利视频| 三妻四妾完整版在线观看电视剧| 影音先锋日韩精品| 91成人免费网站| 婷婷电影在线观看| 免费不卡视频| 欧美人在线观看| 青青青爽久久午夜综合久久午夜| 先锋在线资源一区二区三区| 岛国av免费在线观看| 成人免费毛片aaaaa**| 青春草在线免费视频| 亚洲第一网站免费视频| 激情网址大全| 午夜精品婷婷| 小说区图片区色综合区| 国产精品夜夜夜一区二区三区尤| 亚洲一区二区在线观看视频| 中文字幕一区二区三区乱码不卡| 亚洲美女少妇无套啪啪呻吟| 欧美日韩亚洲国内综合网| 日韩成人中文字幕| 4444亚洲人成无码网在线观看| 国内免费精品永久在线视频| 日韩成人视屏| av网站免费在线| 日本美女一区| 久久动漫网址| 手机成人在线| 日韩色av导航| 日韩一区二区三区av| 亚洲综合日韩| 免费a在线看| 中文字幕人成高清视频| 亚洲国产精品综合久久久| 自拍偷拍校园春色| 中文字幕在线2018| 免费看成人午夜电影| 国内精品久久久| 五月天丁香在线| 91视频免费在线看| 国产一卡2卡3卡4卡网站免费| 成人影视亚洲图片在线| 久久久一区二区三区捆绑**| 少妇一级黄色片| 涩涩视频在线观看| 97精品视频在线播放| 久久精品水蜜桃av综合天堂| 免费日本一区二区三区视频| 亚洲品质自拍视频| 黑人中文字幕一区二区三区| 欧美精品在线视频观看| 伊人手机在线视频| 成人免费黄色网址| 亚洲一区二区三区四区五区黄| 久久精品国产亚洲av香蕉| 久久久欧美一区二区| 国产成人亚洲精品自产在线| 国产一区二区高清在线| 日韩av网站导航| 日韩国产精品91| 成人网18免费软件大全| 成人免费黄色网址| 亚洲一二三区视频在线观看| 欧美 日韩 国产一区| 一区视频在线看| 欧美日韩在线不卡一区| 欧美在线视频免费观看| 国产精品传媒在线观看| 天天操夜夜操视频| 日韩欧美国产wwwww| 亚洲欧美激情在线视频| 亚洲一区色图| 亚洲综合色站| eeuss国产一区二区三区四区| 99久久综合国产精品| 激情综合在线| 中文字幕在线免费| 亚洲女爱视频在线| 亚洲 欧美 日韩 国产综合 在线| 欧美日韩一区 二区 三区 久久精品| 91在线视频播放地址| 欧美三级韩国三级日本三斤| 亚洲精品美女久久7777777| 亚洲a v网站| 四虎成人免费在线| 青青草原综合久久大伊人精品| 天天爽人人爽夜夜爽| 91国内在线视频| 国产精品少妇在线视频| www.精品国产| 亚洲精品中文字幕有码专区| 九九精品九九| 欧美xxxxxxxxx| 日韩视频免费观看高清在线视频| 日韩精品久久久久久久电影99爱| 国产九九在线| 久久久久久久久久久免费| 日韩精品一区二区三区三区免费| videoxxxx另类日本极品| 日韩久久精品一区| 国产精品免费久久久久| 欧美成人免费全部观看天天性色| 人与牲动交xxxxbbbb高清| 91九色在线观看| 免费a在线观看播放| 在线成人激情黄色| 国产在线拍揄自揄拍| 熟女av一区二区| 一区二区三区波多野结衣在线观看| 日韩黄色在线播放| 国产精品久久久久9999高清| 亚洲精品久久久久中文字幕欢迎你| 小草av在线播放| 亚洲免费大片在线观看| 91在线免费看网站| 精品久久国产字幕高潮| 日韩成人av影视| 亚洲综合久久av一区二区三区| 亚洲国产精品成人一区二区在线| 国产调教打屁股xxxx网站| 国产性做久久久久久| 日韩毛片一二三区| 嫩草成人www欧美| 色国产精品一区在线观看| 香蕉久久夜色精品国产更新时间| 成人黄色国产精品网站大全在线免费观看| 人人在线视频| 欧美精品生活片| 久久国产剧场电影| 天天操,天天操| 影音先锋在线资源中文字幕| 祥仔av免费一区二区三区四区| 开心久久婷婷综合中文字幕| 91插插插插插插| 成人黄色在线观看视频| 国产白丝精品91爽爽久久| 亚洲一区二区不卡视频| 91精品免费在线观看| 成人国产精品一区二区| 肉肉av福利一精品导航| 性网站在线播放| 亚洲妇女av| 99riav国产精品| 99久久一区二区| 久久久亚洲精品石原莉奈| 一本色道久久亚洲综合精品蜜桃| 51国偷自产一区二区三区的来源| 女人又爽又黄免费女仆| 中文字幕在线亚洲精品|