欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)-創(chuàng)新互聯(lián)

前言

先簡單說下本次的主題,由于我最近做的是物聯(lián)網(wǎng)相關(guān)的開發(fā)工作,其中就不免會遇到和設(shè)備的交互。

創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供泉港網(wǎng)站建設(shè)、泉港做網(wǎng)站、泉港網(wǎng)站設(shè)計、泉港網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計與制作、泉港企業(yè)網(wǎng)站模板建站服務(wù),10余年泉港做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡(luò)服務(wù)。

最主要的工作就是要有一個系統(tǒng)來支持設(shè)備的接入、向設(shè)備推送消息;同時還得滿足大量設(shè)備接入的需求。

所以本次分享的內(nèi)容不但可以滿足物聯(lián)網(wǎng)領(lǐng)域同時還支持以下場景:

  • 基于 WEB 的聊天系統(tǒng)(點對點、群聊)。
  • WEB 應(yīng)用中需求服務(wù)端推送的場景。
  • 基于 SDK 的消息推送平臺。

技術(shù)選型

要滿足大量的連接數(shù)、同時支持雙全工通信,并且性能也得有保障。

在 Java 技術(shù)棧中進行選型首先自然是排除掉了傳統(tǒng) IO。

那就只有選 NIO 了,在這個層面其實選擇也不多,考慮到社區(qū)、資料維護等方面最終選擇了 Netty。

最終的架構(gòu)圖如下:

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

現(xiàn)在看著蒙沒關(guān)系,下文一一介紹。

協(xié)議解析

既然是一個消息系統(tǒng),那自然得和客戶端定義好雙方的協(xié)議格式。

常見和簡單的是 HTTP 協(xié)議,但我們的需求中有一項需要是雙全工的交互方式,同時 HTTP 更多的是服務(wù)于瀏覽器。我們需要的是一個更加精簡的協(xié)議,減少許多不必要的數(shù)據(jù)傳輸。

因此我覺得最好是在滿足業(yè)務(wù)需求的情況下定制自己的私有協(xié)議,在我這個場景下其實有標(biāo)準(zhǔn)的物聯(lián)網(wǎng)協(xié)議。

如果是其他場景可以借鑒現(xiàn)在流行的 RPC 框架定制私有協(xié)議,使得雙方通信更加高效。

不過根據(jù)這段時間的經(jīng)驗來看,不管是哪種方式都得在協(xié)議中預(yù)留安全相關(guān)的位置。

協(xié)議相關(guān)的內(nèi)容就不過討論了,更多介紹具體的應(yīng)用。

簡單實現(xiàn)

首先考慮如何實現(xiàn)功能,再來思考百萬連接的情況。

注冊鑒權(quán)

在做真正的消息上、下行之前首先要考慮的就是鑒權(quán)問題。

就像你使用微信一樣,第一步怎么也得是登錄吧,不能無論是誰都可以直接連接到平臺。

所以第一步得是注冊才行。

如上面架構(gòu)圖中的 注冊/鑒權(quán) 模塊。通常來說都需要客戶端通過 HTTP 請求傳遞一個唯一標(biāo)識,后臺鑒權(quán)通過之后會響應(yīng)一個 token,并將這個 token 和客戶端的關(guān)系維護到 Redis 或者是 DB 中。

客戶端將這個 token 也保存到本地,今后的每一次請求都得帶上這個 token。一旦這個 token 過期,客戶端需要再次請求獲取 token。

鑒權(quán)通過之后客戶端會直接通過 TCP長連接到圖中的 push-server 模塊。

這個模塊就是真正處理消息的上、下行。

保存通道關(guān)系

在連接接入之后,真正處理業(yè)務(wù)之前需要將當(dāng)前的客戶端和 Channel 的關(guān)系維護起來。

假設(shè)客戶端的唯一標(biāo)識是手機號碼,那就需要把手機號碼和當(dāng)前的 Channel 維護到一個 Map 中。

這點和之前 SpringBoot 整合長連接心跳機制 類似。

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

同時為了可以通過 Channel 獲取到客戶端唯一標(biāo)識(手機號碼),還需要在 Channel 中設(shè)置對應(yīng)的屬性:

public

static

void
 putClientId
(
Channel
 channel
,

String
 clientId
)

{
 channel
.
attr
(
CLIENT_ID
).
set
(
clientId
);
}

獲取時手機號碼時:

public

static

String
 getClientId
(
Channel
 channel
)

{

return

(
String
)
getAttribute
(
channel
,
 CLIENT_ID
);
}

這樣當(dāng)我們客戶端下線的時便可以記錄相關(guān)日志:

String
 telNo 
=

NettyAttrUtil
.
getClientId
(
ctx
.
channel
());
NettySocketHolder
.
remove
(
telNo
);
log
.
info
(
"客戶端下線,TelNo="

+
 telNo
);

這里有一點需要注意:存放客戶端與 Channel 關(guān)系的 Map 最好是預(yù)設(shè)好大?。ū苊饨?jīng)常擴容),因為它將是使用最為頻繁同時也是占用內(nèi)存大的一個對象。

消息上行

接下來則是真正的業(yè)務(wù)數(shù)據(jù)上傳,通常來說第一步是需要判斷上傳消息輸入什么業(yè)務(wù)類型。

在聊天場景中,有可能上傳的是文本、圖片、視頻等內(nèi)容。

所以我們得進行區(qū)分,來做不同的處理;這就和客戶端協(xié)商的協(xié)議有關(guān)了。

  • 可以利用消息頭中的某個字段進行區(qū)分。
  • 更簡單的就是一個 JSON 消息,拿出一個字段用于區(qū)分不同消息。

不管是哪種只有可以區(qū)分出來即可。

消息解析與業(yè)務(wù)解耦

消息可以解析之后便是處理業(yè)務(wù),比如可以是寫入數(shù)據(jù)庫、調(diào)用其他接口等。

我們都知道在 Netty 中處理消息一般是在 channelRead() 方法中。

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

在這里可以解析消息,區(qū)分類型。

但如果我們的業(yè)務(wù)邏輯也寫在里面,那這里的內(nèi)容將是巨多無比。

甚至我們分為好幾個開發(fā)來處理不同的業(yè)務(wù),這樣將會出現(xiàn)許多沖突、難以維護等問題。

所以非常有必要將消息解析與業(yè)務(wù)處理完全分離開來。

這時面向接口編程就發(fā)揮作用了。

這里的核心代碼和 「造個輪子」——cicada(輕量級 WEB 框架) 是一致的。

都是先定義一個接口用于處理業(yè)務(wù)邏輯,然后在解析消息之后通過反射創(chuàng)建具體的對象執(zhí)行其中的 處理函數(shù)即可。

這樣不同的業(yè)務(wù)、不同的開發(fā)人員只需要實現(xiàn)這個接口同時實現(xiàn)自己的業(yè)務(wù)邏輯即可。

偽代碼如下:

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

上行還有一點需要注意;由于是基于長連接,所以客戶端需要定期發(fā)送心跳包用于維護本次連接。同時服務(wù)端也會有相應(yīng)的檢查,N 個時間間隔沒有收到消息之后將會主動斷開連接節(jié)省資源。

這點使用一個 IdleStateHandler 就可實現(xiàn),更多內(nèi)容可以查看 Netty(一) SpringBoot 整合長連接心跳機制。

消息下行

有了上行自然也有下行。比如在聊天的場景中,有兩個客戶端連上了 push-server,他們直接需要點對點通信。

這時的流程是:

  • A 將消息發(fā)送給服務(wù)器。
  • 服務(wù)器收到消息之后,得知消息是要發(fā)送給 B,需要在內(nèi)存中找到 B 的 Channel。
  • 通過 B 的 Channel 將 A 的消息轉(zhuǎn)發(fā)下去。

這就是一個下行的流程。

甚至管理員需要給所有在線用戶發(fā)送系統(tǒng)通知也是類似:

遍歷保存通道關(guān)系的 Map,挨個發(fā)送消息即可。這也是之前需要存放到 Map 中的主要原因。

偽代碼如下:

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

分布式方案

單機版的實現(xiàn)了,現(xiàn)在著重講講如何實現(xiàn)百萬連接。

百萬連接其實只是一個形容詞,更多的是想表達(dá)如何來實現(xiàn)一個分布式的方案,可以靈活的水平拓展從而能支持更多的連接。

再做這個事前首先得搞清楚我們單機版的能支持多少連接。影響這個的因素就比較多了。

  • 服務(wù)器自身配置。內(nèi)存、CPU、網(wǎng)卡、Linux 支持的大文件打開數(shù)等。
  • 應(yīng)用自身配置,因為 Netty 本身需要依賴于堆外內(nèi)存,但是 JVM 本身也是需要占用一部分內(nèi)存的,比如存放通道關(guān)系的大 Map。這點需要結(jié)合自身情況進行調(diào)整。

結(jié)合以上的情況可以測試出單個節(jié)點能支持的大連接數(shù)。

單機無論怎么優(yōu)化都是有上限的,這也是分布式主要解決的問題。

架構(gòu)介紹

在將具體實現(xiàn)之前首先得講講上文貼出的整體架構(gòu)圖。

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

先從左邊開始。

上文提到的 注冊鑒權(quán) 模塊也是集群部署的,通過前置的 Nginx 進行負(fù)載。之前也提過了它主要的目的是來做鑒權(quán)并返回一個 token 給客戶端。

但是 push-server 集群之后它又多了一個作用。那就是得返回一臺可供當(dāng)前客戶端使用的 push-server。

右側(cè)的 平臺 一般指管理平臺,它可以查看當(dāng)前的實時在線數(shù)、給指定客戶端推送消息等。

推送消息則需要經(jīng)過一個推送路由( push-server)找到真正的推送節(jié)點。

其余的中間件如:Redis、Zookeeper、Kafka、MySQL 都是為了這些功能所準(zhǔn)備的,具體看下面的實現(xiàn)。

注冊發(fā)現(xiàn)

首先第一個問題則是 注冊發(fā)現(xiàn), push-server 變?yōu)槎嗯_之后如何給客戶端選擇一臺可用的節(jié)點是第一個需要解決的。

這塊的內(nèi)容其實已經(jīng)在 分布式(一) 搞定服務(wù)注冊與發(fā)現(xiàn) 中詳細(xì)講過了。

所有的 push-server 在啟動時候需要將自身的信息注冊到 Zookeeper 中。

注冊鑒權(quán) 模塊會訂閱 Zookeeper 中的節(jié)點,從而可以獲取最新的服務(wù)列表。結(jié)構(gòu)如下:

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

以下是一些偽代碼:

應(yīng)用啟動注冊 Zookeeper。

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

對于 注冊鑒權(quán)模塊來說只需要訂閱這個 Zookeeper 節(jié)點:

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

路由策略

既然能獲取到所有的服務(wù)列表,那如何選擇一臺剛好合適的 push-server 給客戶端使用呢?

這個過程重點要考慮以下幾點:

  • 盡量保證各個節(jié)點的連接均勻。
  • 增刪節(jié)點是否要做 Rebalance。

首先保證均衡有以下幾種算法:

  • 輪詢。挨個將各個節(jié)點分配給客戶端。但會出現(xiàn)新增節(jié)點分配不均勻的情況。
  • Hash 取模的方式。類似于 HashMap,但也會出現(xiàn)輪詢的問題。當(dāng)然也可以像 HashMap 那樣做一次 Rebalance,讓所有的客戶端重新連接。不過這樣會導(dǎo)致所有的連接出現(xiàn)中斷重連,代價有點大。
  • 由于 Hash 取模方式的問題帶來了 一致性Hash算法,但依然會有一部分的客戶端需要 Rebalance。
  • 權(quán)重??梢允謩诱{(diào)整各個節(jié)點的負(fù)載情況,甚至可以做成自動的,基于監(jiān)控當(dāng)某些節(jié)點負(fù)載較高就自動調(diào)低權(quán)重,負(fù)載較低的可以提高權(quán)重。

還有一個問題是:

當(dāng)我們在重啟部分應(yīng)用進行升級時,在該節(jié)點上的客戶端怎么處理?

由于我們有心跳機制,當(dāng)心跳不通之后就可以認(rèn)為該節(jié)點出現(xiàn)問題了。那就得重新請求 注冊鑒權(quán)模塊獲取一個可用的節(jié)點。在弱網(wǎng)情況下同樣適用。

如果這時客戶端正在發(fā)送消息,則需要將消息保存到本地等待獲取到新的節(jié)點之后再次發(fā)送。

有狀態(tài)連接

在這樣的場景中不像是 HTTP 那樣是無狀態(tài)的,我們得明確的知道各個客戶端和連接的關(guān)系。

在上文的單機版中我們將這個關(guān)系保存到本地的緩存中,但在分布式環(huán)境中顯然行不通了。

比如在平臺向客戶端推送消息的時候,它得首先知道這個客戶端的通道保存在哪臺節(jié)點上。

借助我們以前的經(jīng)驗,這樣的問題自然得引入一個第三方中間件用來存放這個關(guān)系。

也就是架構(gòu)圖中的存放 路由關(guān)系的Redis,在客戶端接入 push-server 時需要將當(dāng)前客戶端唯一標(biāo)識和服務(wù)節(jié)點的 ip+port 存進 Redis。

同時在客戶端下線時候得在 Redis 中刪掉這個連接關(guān)系。

這樣在理想情況下各個節(jié)點內(nèi)存中的 map 關(guān)系加起來應(yīng)該正好等于 Redis 中的數(shù)據(jù)。

偽代碼如下:

一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)

這里存放路由關(guān)系的時候會有并發(fā)問題,最好是換為一個 lua 腳本。

推送路由

設(shè)想這樣一個場景:管理員需要給最近注冊的客戶端推送一個系統(tǒng)消息會怎么做?

結(jié)合架構(gòu)圖

假設(shè)這批客戶端有 10W 個,首先我們需要將這批號碼通過 平臺下的 Nginx 下發(fā)到一個推送路由中。

為了提高效率甚至可以將這批號碼再次分散到每個 push-route 中。

拿到具體號碼之后再根據(jù)號碼的數(shù)量啟動多線程的方式去之前的路由 Redis 中獲取客戶端所對應(yīng)的 push-server。

再通過 HTTP 的方式調(diào)用 push-server 進行真正的消息下發(fā)(Netty 也很好的支持 HTTP 協(xié)議)。

推送成功之后需要將結(jié)果更新到數(shù)據(jù)庫中,不在線的客戶端可以根據(jù)業(yè)務(wù)再次推送等。

消息流轉(zhuǎn)

也許有些場景對于客戶端上行的消息非??粗?,需要做持久化,并且消息量非常大。

在 push-sever 做業(yè)務(wù)顯然不合適,這時完全可以選擇 Kafka 來解耦。

將所有上行的數(shù)據(jù)直接往 Kafka 里丟后就不管了。

再由消費程序?qū)?shù)據(jù)取出寫入數(shù)據(jù)庫中即可。

其實這塊內(nèi)容也很值得討論,可以先看這篇了解下:強如 Disruptor 也發(fā)生內(nèi)存溢出?

后續(xù)談到 Kafka 再做詳細(xì)介紹。

分布式問題

分布式解決了性能問題但卻帶來了其他麻煩。

應(yīng)用監(jiān)控

比如如何知道線上幾十個 push-server 節(jié)點的健康狀況?

這時就得監(jiān)控系統(tǒng)發(fā)揮作用了,我們需要知道各個節(jié)點當(dāng)前的內(nèi)存使用情況、GC。

以及操作系統(tǒng)本身的內(nèi)存使用,畢竟 Netty 大量使用了堆外內(nèi)存。

同時需要監(jiān)控各個節(jié)點當(dāng)前的在線數(shù),以及 Redis 中的在線數(shù)。理論上這兩個數(shù)應(yīng)該是相等的。

這樣也可以知道系統(tǒng)的使用情況,可以靈活的維護這些節(jié)點數(shù)量。

日志處理

日志記錄也變得異常重要了,比如哪天反饋有個客戶端一直連不上,你得知道問題出在哪里。

最好是給每次請求都加上一個 traceID 記錄日志,這樣就可以通過這個日志在各個節(jié)點中查看到底是卡在了哪里。

以及 ELK 這些工具都得用起來才行。

總結(jié)

本次是結(jié)合我日常經(jīng)驗得出的,有些坑可能在工作中并沒有踩到,所有還會有一些遺漏的地方。

就目前來看想做一個穩(wěn)定的推送系統(tǒng)其實是比較麻煩的,其中涉及到的點非常多,只有真正做過之后才會知道。

創(chuàng)新互聯(lián)www.cdcxhl.cn,專業(yè)提供香港、美國云服務(wù)器,動態(tài)BGP最優(yōu)骨干路由自動選擇,持續(xù)穩(wěn)定高效的網(wǎng)絡(luò)助力業(yè)務(wù)部署。公司持有工信部辦法的idc、isp許可證, 機房獨有T級流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確進行流量調(diào)度,確保服務(wù)器高可用性。佳節(jié)活動現(xiàn)已開啟,新人活動云服務(wù)器買多久送多久。

分享文章:一篇文章教你如何設(shè)計一個百萬級的消息推送系統(tǒng)-創(chuàng)新互聯(lián)
路徑分享:http://aaarwkj.com/article46/deodeg.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站設(shè)計、用戶體驗商城網(wǎng)站、搜索引擎優(yōu)化、網(wǎng)站維護、Google

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站網(wǎng)頁設(shè)計
日韩欧美亚洲国产每日更新| 人妻日韩字幕一区二区| 亚洲成人av日韩在线| 午夜香蕉av一区二区三区| 午夜福利大片在线观看视频| 亚洲精品一区二区三区中文字幕| 国产日韩视频一区二区| 国产精品国产三级丝袜| 亚洲av优选在线观看精品| 国产97成人亚洲综合在线| 亚洲国产精品久久久精品| 亚洲中文字幕少妇视频| 国产三级在线视频不卡| 日本高清免费中文字幕| 亚洲最大午夜福利视频| 91久久精品国产免费一区| 欧美激情欧美精品欧美色浮| 亚洲国产日韩欧美视频| 成人欧美精品一区二区不卡| 国产女同互慰一区二区| 婷婷亚洲五月伊人91| 国产原创剧情av网址| 高颜值紧身牛仔裤国产精品| 国产精品一区巨乳人妻| 麻豆色视频在线观看免费| 免费在线黄色生活大片| 亚洲专区综合红桃av| 国产成人亚洲精品乱码| 给我搜一个一级黄色片| 国产精品一区二区三区在线| 国产爆操美女在线观看| 激情视频一区二区三区| 日韩国产欧美色资源在线| 亚洲欧美午夜福利视频| 亚洲免费av一区在线观看| 日韩一区欧美中文字幕| 久久综合午夜福利视频| 欧美亚洲另类在线日韩国产| 亚洲国产精品中文字幕久久| 中文字幕av二区三区人妻| 久久热久久热在线视频|