Tomcat 實現(xiàn)WebSocket詳細(xì)介紹
Tomcat 如何實現(xiàn)WebSocket
WebSocket協(xié)議屬于HTML5標(biāo)準(zhǔn),越來越多瀏覽器已經(jīng)原生支持WebSocket,它能讓客戶端和服務(wù)端實現(xiàn)雙向通信。在客戶端和服務(wù)器端建立一條WebSocket連接后,服務(wù)器端消息可直接發(fā)送到客戶端,從而打破傳統(tǒng)的請求響應(yīng)模式,避免了無意義的請求。比如傳統(tǒng)的方式可能會使用AJAX不斷請求服務(wù)器端,而WebSocket則可以直接發(fā)送數(shù)據(jù)到客戶端且客戶端不必請求。同時,由于有了瀏覽器的原生支持,編寫客戶端應(yīng)用程序也變得更加便捷且不必依賴第三方插件。另外,WebSocket協(xié)議摒棄了HTTP協(xié)議繁瑣的請求頭,而是以數(shù)據(jù)幀的方式進(jìn)行傳輸,效率更高。
圖為WebSocket協(xié)議通信的過程,首先客戶端會發(fā)送一個握手包告訴服務(wù)器端我想升級成WebSocket,不知道你服務(wù)器端是否同意,這時如果服務(wù)器端支持WebSocket協(xié)議則會返回一個握手包告訴客戶端沒問題,升級已確認(rèn)。然后就成功建立起了一條WebSocket連接,該連接支持雙向通信,并且使用WebSocket協(xié)議的數(shù)據(jù)幀格式發(fā)送消息。

握手過程需要說明下,為了讓W(xué)ebSocket協(xié)議能和現(xiàn)有HTTP協(xié)議Web架構(gòu)互相兼容,所以WebSocket協(xié)議的握手要基于HTTP協(xié)議,比如客戶端會發(fā)送類似如下的HTTP報文到服務(wù)器端請求升級為WebSocket協(xié)議,其中包含的Upgrade: websocket就是告訴服務(wù)器端我想升級協(xié)議:
GET ws://localhost:8080/hello HTTP/1.1 Origin: http://localhost:8080 Connection: Upgrade Host: localhost:8080 Sec-WebSocket-Key: uRovscZjNol/umbTt5uKmw== Upgrade: websocket Sec-WebSocket-Version: 13
此時如果服務(wù)器端支持WebSocket協(xié)議,則它會發(fā)送一個同意客戶端升級協(xié)議的報文,具體報文類似如下,其中Upgrade: websocket就是告訴客戶端我同意你升級協(xié)議:
HTTP/1.1 101 WebSocket Protocol Handshake Date: Fri, 10 Feb 2016 17:38:18 GMT Connection: Upgrade Server: Kaazing Gateway Upgrade: WebSocket Sec-WebSocket-Accept: rLHCkw/SKsO9GAH/ZSFhBATDKrU=
完成如上握手后,HTTP協(xié)議連接就被打破,接下去則是開始使用WebSocket協(xié)議進(jìn)行雙方通信,這條連接還是原來的那條TCP/IP連接,端口也還是原來的80或443。
下面舉一個Tomcat中編寫WebSocket的簡單例子:
public class HelloWebSocketServlet extends WebSocketServlet {
private static List<MessageInbound> socketList = new ArrayList<MessageInbound>();
protected StreamInbound createWebSocketInbound(String subProtocol,HttpServletRequest request){
return new WebSocketMessageInbound();
}
public class WebSocketMessageInbound extends MessageInbound{
protected void onClose(int status){
super.onClose(status);
socketList.remove(this);
}
protected void onOpen(WsOutbound outbound){
super.onOpen(outbound);
socketList.add(this);
}
@Override
protected void onBinaryMessage(ByteBuffer message) throws IOException {
}
@Override
protected void onTextMessage(CharBuffer message) throws IOException {
for(MessageInbound messageInbound : socketList){
CharBuffer buffer = CharBuffer.wrap(message);
WsOutbound outbound = messageInbound.getWsOutbound();
outbound.writeTextMessage(buffer);
outbound.flush();
}
}
}
}
這個Servlet必須要繼承WebSocketServlet,接著創(chuàng)建一個繼承MessageInbound的WebSocketMessageInbound類,在該類中填充onClose、onOpen、onBinaryMessage和onTextMessage等方法即可完成各個事件的邏輯,其中onOpen會在一個WebSocket連接建立時被調(diào)用,onClose會在一個WebSocket關(guān)閉時被調(diào)用,onBinaryMessage則是Binary方式下接收到客戶端數(shù)據(jù)時被調(diào)用,onTextMessage則是Text方式下接收到客戶端數(shù)據(jù)時被調(diào)用。上面一段代碼實現(xiàn)了一個廣播的效果。
按照上面的處理邏輯,Tomcat對WebSocket的集成就不會太難了,就是在處理請求時如果遇到WebSocket協(xié)議請求則做特殊處理,保持住連接并在適當(dāng)?shù)臅r機(jī)調(diào)用WebSocketServlet的MessageInbound的onClose、onOpen、onBinaryMessage和onTextMessage等方法。由于WebSocket一般建議在NIO模式下使用,所以看看NIO模式集成WebSocket協(xié)議。
如圖,對于WebSocket的客戶端連接被接收器接收后注冊到NioChannel隊列中,Poller組件不斷輪休是否有NioChannel需要處理,如果有則經(jīng)過處理管道后進(jìn)到繼承了WebSocketServlet的Servlet上,WebSocketServlet的doGet方法會處理WebSocket握手,告訴返回客戶端同意升級協(xié)議。往后Poller繼續(xù)不斷輪休相關(guān)NioChannel,一旦發(fā)現(xiàn)是使用WebSocket協(xié)議的管道則會調(diào)用MessageInbound的相關(guān)方法,完成不同事件的處理,從而實現(xiàn)對WebSocket協(xié)議的支持。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
- IIS和tomcat5多站點配置流程
- win2000server IIS和tomcat5多站點配置
- WIN2003上Apache2+IIS6+Tomcat5之多站點完美配置篇
- Tomcat 發(fā)布程序使用cmd查看端口占用、相應(yīng)進(jìn)程、殺死進(jìn)程等的命令
- Tomcat日志文件定時清理備份的腳本
- 詳解Ngigx+Tomcat配置動靜分離,負(fù)載均衡
- 詳解CentOS安裝tomcat并且部署Java Web項目
- linux下實時查看tomcat運行日志的方法
- 在同一臺服務(wù)器上配置多個Tomcat的方法
- Docker 自動部署tomcat詳解
- Shell腳本切割tomcat的日志文件
- Tomcat 多站點配置詳解及實現(xiàn)方法
相關(guān)文章
使用jaxp進(jìn)行dom解析_動力節(jié)點Java學(xué)院整理
這篇文章主要介紹了使用jaxp進(jìn)行dom解析的相關(guān)資料,需要的朋友可以參考下2017-08-08
spring boot啟動后直接關(guān)閉了的問題解決
本文主要介紹了spring boot啟動后直接關(guān)閉了的問題解決,SpringBoot項目啟動后自動關(guān)閉的原因是未引入web依賴,導(dǎo)致以普通Java項目運行,下面就來介紹一下解決方法,感興趣的可以了解一下2025-02-02
java如何根據(jù)提供word模板導(dǎo)出word文檔詳解
在日常的開發(fā)工作中,我們時常會遇到導(dǎo)出Word文檔報表的需求,比如公司的財務(wù)報表、醫(yī)院的患者統(tǒng)計報表、電商平臺的銷售報表等等,這篇文章主要給大家介紹了關(guān)于java如何根據(jù)提供word模板導(dǎo)出word文檔的相關(guān)資料,需要的朋友可以參考下2023-09-09
Java中Easyexcel?實現(xiàn)批量插入圖片功能
這篇文章主要介紹了Easyexcel?實現(xiàn)批量插入圖片,本文通過實例代碼給大家介紹了easyexcel文檔處理工具、自定義圖片處理器的相關(guān)知識,需要的朋友可以參考下2022-04-04
Java?中的整型數(shù)據(jù)類型((byte,?short,?int,?long?)的取值范圍及使用不同的整型的場景
Java中的整型數(shù)據(jù)類型包括byte、short、int和long,每種類型都有不同的內(nèi)存占用和取值范圍,選擇合適的整型類型取決于具體需求,如內(nèi)存敏感場景、性能考量和避免整數(shù)溢出,示例代碼展示了如何聲明和初始化不同整型變量,并打印它們的最大值,感興趣的朋友一起看看吧2025-03-03
Java?深入學(xué)習(xí)static關(guān)鍵字和靜態(tài)屬性及方法
這篇文章主要介紹了Java?深入學(xué)習(xí)static關(guān)鍵字和靜態(tài)屬性及方法,文章通過圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-09-09
SpringBoot配置ShedLock分布式定時任務(wù)
ShedLock是一個在分布式環(huán)境中使用的定時任務(wù)框架,這篇文章主要介紹了SpringBoot配置ShedLock分布式定時任務(wù),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05

