偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

用OkHttp實現(xiàn)WebSocket長連接

開發(fā) 前端
先簡單介紹下WebSocket。我們都知道Http是處于應用層的一個通信協(xié)議,但是只支持單向主動通信,做不到服務器主動向客戶端推送消息。而且Http是無狀態(tài)的,即每次通信都沒有關聯(lián)性,導致跟服務器關系不緊密。

今天給大家?guī)硪黄衔恼?,介紹WebSocket,大家可以了解了解。

前言

最近老板又來新需求了,要做一個物聯(lián)網(wǎng)相關的app,其中有個需求是客戶端需要收發(fā)服務器不定期發(fā)出的消息。

內(nèi)心OS:

  • 這咋整呢?通過接口輪詢?定時訪問接口,有數(shù)據(jù)就更新?
  • 不行不行,這樣浪費資源了,還耗電,會導致很多請求都是無效的網(wǎng)絡操作。
  • 那就長連接唄?WebSocket協(xié)議好像不錯,通過握手建立長連接后,可以隨時收發(fā)服務器的消息。那就它了!
  • 怎么集成呢?正好前段時間復習OkHttp源碼的時候發(fā)現(xiàn)了它是支持Websocket協(xié)議的,那就用它試試吧!

開淦!

WebSocket介紹

先簡單介紹下WebSocket。我們都知道Http是處于應用層的一個通信協(xié)議,但是只支持單向主動通信,做不到服務器主動向客戶端推送消息。而且Http是無狀態(tài)的,即每次通信都沒有關聯(lián)性,導致跟服務器關系不緊密。

為了解決和服務器長時間通信的痛點呢,HTML5規(guī)范引出了WebSocket協(xié)議(知道這名字咋來的吧,人家HTML5規(guī)范引出的,隨爸姓),是一種建立在TCP協(xié)議基礎上的全雙工通信的協(xié)議。他跟Http同屬于應用層協(xié)議,下層還是需要通過TCP建立連接。

但是,WebSocket在TCP連接建立后,還要通過Http進行一次握手,也就是通過Http發(fā)送一條GET請求消息給服務器,告訴服務器我要建立WebSocket連接了,你準備好哦,具體做法就是在頭部信息中添加相關參數(shù)。然后服務器響應我知道了,并且將連接協(xié)議改成WebSocket,開始建立長連接。

這里貼上請求頭和響應頭信息,從網(wǎng)上找了一張圖:

3851594110877_.pic.jpg

 

簡單說明下參數(shù):

  • URL一般是以ws或者wss開頭,ws對應Websocket協(xié)議,wss對應在TLS之上的WebSocket。類似于Http和Https的關系。
  • 請求方法為GET方法。
  • Connection:Upgrade,表示客戶端要連接升級,不用Http協(xié)議。
  • Upgrade:websocket, 表示客戶端要升級建立Websocket連接。
  • Sec-Websocket-Key:key, 這個key是隨機生成的,服務器會通過這個參數(shù)驗證該請求是否有效。
  • Sec-WebSocket-Version:13, websocket使用的協(xié)議,一般就是13。
  • Sec-webSocket-Extension:permessage-deflate,客戶端指定的一些擴展協(xié)議,比如這里permessage-deflate就是WebSocket的一種壓縮協(xié)議。
  • 響應碼101,表示響應協(xié)議升級,后續(xù)的數(shù)據(jù)交互都按照Upgradet指定的WebSocket協(xié)議來。

OkHttp實現(xiàn)

添加OkHttp依賴

  1. implementation("com.squareup.okhttp3:okhttp:4.7.2"

實現(xiàn)代碼

首先是初始化OkHttpClient和WebSocket實例:

  1. /** 
  2.  * 初始化WebSocket 
  3.  */ 
  4. public void init() { 
  5.     mWbSocketUrl = "ws://echo.websocket.org"
  6.     mClient = new OkHttpClient.Builder() 
  7.             .pingInterval(10, TimeUnit.SECONDS) 
  8.             .build(); 
  9.     Request request = new Request.Builder() 
  10.             .url(mWbSocketUrl) 
  11.             .build(); 
  12.     mWebSocket = mClient.newWebSocket(request, new WsListener()); 

這里主要是配置了OkHttp的一些參數(shù),以及WebSocket的連接地址。其中newWebSocket方法就是進行WebSocket的初始化和連接。

這里要注意的點是pingInterval方法的配置,這個方法主要是用來設置WebSocket連接的?;?。相信做過長連接的同學都知道,一個長連接一般要隔幾秒發(fā)送一條消息告訴服務器我在線,而服務器也會回復一個消息表示收到了,這樣就確認了連接正常,客戶端和服務器端都在線。

如果服務器沒有按時收到這個消息那么服務器可能就會主動關閉這個連接,節(jié)約資源。客戶端沒有正常收到這個返回的消息,也會做一些類似重連的操作,所以這個保活消息非常重要。

我們稱這個消息叫作心跳包,一般用PING,PONG表示,像乒乓球一樣,一來一回。所以這里的pingInterval就是設置心跳包發(fā)送的間隔時間,設置了這個方法之后,OkHttp就會自動幫我們發(fā)送心跳包事件,也就是ping包。當間隔時間到了,沒有收到pong包的話,監(jiān)聽事件中的onFailure方法就會被調(diào)用,此時我們就可以進行重連。

但是由于實際業(yè)務需求不一樣,以及okhttp中心跳包事件給予我們權限較少,所以我們也可以自己完成心跳包事件,即在WebSocket連接成功之后,開始定時發(fā)送ping包,在下一次發(fā)送ping包之前檢查上一個pong包是否收到,如果沒收到,就視為異常,開始重連。感興趣的同學可以看看文末的相關源碼。

建立連接后,我們就可以正常發(fā)送和讀取消息了,也就是在上文WsListener監(jiān)聽事件中表現(xiàn):

  1. //監(jiān)聽事件,用于收消息,監(jiān)聽連接的狀態(tài) 
  2. class WsListener extends WebSocketListener { 
  3.     @Override 
  4.     public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { 
  5.         super.onClosed(webSocket, code, reason); 
  6.     } 
  7.  
  8.     @Override 
  9.     public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { 
  10.         super.onClosing(webSocket, code, reason); 
  11.     } 
  12.  
  13.     @Override 
  14.     public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, @Nullable Response response) { 
  15.         super.onFailure(webSocket, t, response); 
  16.     } 
  17.  
  18.     @Override 
  19.     public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { 
  20.         super.onMessage(webSocket, text); 
  21.         Log.e(TAG, "客戶端收到消息:" + text); 
  22.         onWSDataChanged(DATE_NORMAL, text); 
  23.        //測試發(fā)消息 
  24.         webSocket.send("我是客戶端,你好啊"); 
  25.     } 
  26.  
  27.     @Override 
  28.     public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { 
  29.         super.onMessage(webSocket, bytes); 
  30.     } 
  31.  
  32.     @Override 
  33.     public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { 
  34.         super.onOpen(webSocket, response); 
  35.         Log.e(TAG,"連接成功!"); 
  36.     } 
  37.  
  38.  
  39. //發(fā)送String消息 
  40. public void send(final String message) { 
  41.     if (mWebSocket != null) { 
  42.         mWebSocket.send(message); 
  43.     } 
  44.  
  45. /** 
  46.  * 發(fā)送byte消息 
  47.  * @param message 
  48.  */ 
  49. public void send(final ByteString message) { 
  50.     if (mWebSocket != null) { 
  51.         mWebSocket.send(message); 
  52.     } 
  53. }     
  54.  
  55. //主動斷開連接 
  56. public void disconnect(int code, String reason) { 
  57.     if (mWebSocket != null
  58.         mWebSocket.close(code, reason); 

這里要注意,回調(diào)的方法都是在子線程回調(diào)的,如果需要更新UI,需要切換到主線程。

基本操作就這么多,還是很簡單的吧,初始化Websocket——連接——連接成功——收發(fā)消息。

其中WebSocket類是一個操作接口,主要提供了以下幾個方法

  • send(text: String)發(fā)送一個String類型的消息
  • send(bytes: ByteString) 發(fā)送一個二進制類型的消息
  • close(code: Int, reason: String?)關閉WebSocket連接

如果有同學想測試下WebSocket的功能但是又沒有實際的服務器,怎么辦呢?其實OkHttp官方有一個MockWebSocket服務,可以用來模擬服務端,下面我們一起試一下:

模擬服務器

首先集成MockWebSocket服務庫:

  1. implementation 'com.squareup.okhttp3:mockwebserver:4.7.2' 

然后就可以新建MockWebServer,并加入MockResponse作為接收消息的響應。

  1. MockWebServer mMockWebServer = new MockWebServer(); 
  2.   MockResponse response = new MockResponse() 
  3.           .withWebSocketUpgrade(new WebSocketListener() { 
  4.               @Override 
  5.               public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { 
  6.                   super.onOpen(webSocket, response); 
  7.                   //有客戶端連接時回調(diào) 
  8.                   Log.e(TAG, "服務器收到客戶端連接成功回調(diào):"); 
  9.                   mWebSocket = webSocket; 
  10.                   mWebSocket.send("我是服務器,你好呀"); 
  11.               } 
  12.  
  13.               @Override 
  14.               public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { 
  15.                   super.onMessage(webSocket, text); 
  16.  
  17.                   Log.e(TAG, "服務器收到消息:" + text); 
  18.               } 
  19.  
  20.               @Override 
  21.               public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { 
  22.                   super.onClosed(webSocket, code, reason); 
  23.                   Log.e(TAG, "onClosed:"); 
  24.               } 
  25.           }); 
  26.  
  27.   mMockWebServer.enqueue(response); 

這里服務器端在收到客戶端連接成功消息后,給客戶端發(fā)送了一條消息。要注意的是這段代碼要在子線程執(zhí)行,因為主線程不能進行網(wǎng)絡操作。

然后就可以去初始化Websocket客戶端了:

  1. //獲取連接url,初始化websocket客戶端 
  2. String websocketUrl = "ws://" + mMockWebServer.getHostName() + ":" + mMockWebServer.getPort() + "/"
  3. WSManager.getInstance().init(websocketUrl); 

ok,運行項目

//運行結果 E/jimu: mWbSocketUrl=ws://localhost:38355/ E/jimu: 服務器收到客戶端連接成功回調(diào): E/jimu: 連接成功! E/jimu: 客戶端收到消息:我是服務器,你好呀 E/jimu: 服務器收到消息:我是客戶端,你好啊

參考

https://github.com/square/okhttp

本文轉載自微信公眾號「碼上積木,可以通過以下二維碼關注。轉載本文請聯(lián)系碼上積木公眾號。

 

責任編輯:武曉燕 來源: 碼上積木
相關推薦

2020-03-19 10:13:13

OkHttpWebSocket

2025-01-27 12:31:23

PythonLocustWebSocket

2017-07-11 13:58:10

WebSocket

2021-07-30 15:06:05

鴻蒙HarmonyOS應用

2024-01-11 08:53:58

2021-04-27 18:12:22

WebSocket持久化連接HTTP

2018-06-06 11:01:25

HTTP長連接短連接

2022-10-24 08:14:35

長連接負載均衡Conusmer

2022-12-02 13:49:41

2023-11-28 08:49:01

短輪詢WebSocket長輪詢

2023-01-27 23:31:08

數(shù)據(jù)長輪詢長連接

2020-03-24 15:15:29

HttpClientOkHttpJava

2021-03-24 09:06:01

MySQL長連接短連接

2024-06-06 09:12:40

數(shù)據(jù)庫SQLServer

2015-06-11 10:57:10

2023-12-04 07:14:40

通信微服務

2021-03-25 08:29:33

SpringBootWebSocket即時消息

2023-08-14 08:01:12

websocket8g用戶

2021-11-16 10:45:35

WebSocketWebShellLinux

2020-09-05 17:00:20

HTTP長連接短連接
點贊
收藏

51CTO技術棧公眾號