精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

基于okhttp和RxJava封裝的自動(dòng)重連的WebSocket

系統(tǒng)
RxWebSocket是一個(gè)基于okhttp和RxJava封裝的WebSocket客戶端,此庫(kù)的核心特點(diǎn)是 除了手動(dòng)關(guān)閉WebSocket(就是RxJava取消訂閱),WebSocket在異常關(guān)閉的時(shí)候(onFailure,發(fā)生異常,如WebSocketException等等)。

[[414380]]

想了解更多內(nèi)容,請(qǐng)?jiān)L問(wèn):

51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)

https://harmonyos.51cto.com

一. 概述

RxWebSocket是一個(gè)基于okhttp和RxJava封裝的WebSocket客戶端,此庫(kù)的核心特點(diǎn)是 除了手動(dòng)關(guān)閉WebSocket(就是RxJava取消訂閱),WebSocket在異常關(guān)閉的時(shí)候(onFailure,發(fā)生異常,如WebSocketException等等),會(huì)自動(dòng)重連,永不斷連.其次,對(duì)WebSocket做的緩存處理,同一個(gè)URL,共享一個(gè)WebSocket.

由于是基于RxJava封裝,所以帶來(lái)了無(wú)限可能,可以和RxBinding,Rxlifecycle一起使用,方便對(duì)WebSocket的管理.

效果圖


項(xiàng)目已經(jīng)上傳Jcenter,依賴方法:

  1. //本項(xiàng)目 
  2. compile 'io.github.dzsf:RxWebSocket:1.0.0' 

二. 使用方法

0.初始化,可以也忽略直接使用.

如果你想使用自己的okhttpClient:

  1. OkHttpClient yourClient = new OkHttpClient(); 
  2. RxWebSocketUtil.getInstance().setClient(yourClient); 

 是否打印日志:

  1. RxWebSocketUtil.getInstance().setShowLog(BuildConfig.DEBUG); 

1.獲取一個(gè)WebSocket,接收消息,多種方式:

  1. RxWebSocketUtil.getInstance().getWebSocketInfo(url) 
  2.         .subscribe(new Action1<WebSocketInfo>() { 
  3.             @Override 
  4.             public void call(WebSocketInfo webSocketInfo) { 
  5.                 mWebSocket = webSocketInfo.getWebSocket(); 
  6.                 Log.d("MainActivity", webSocketInfo.getString()); 
  7.                 Log.d("MainActivity""ByteString:" + webSocketInfo.getByteString()); 
  8.             } 
  9.         }); 
  10.  
  11. mWebSocket.send("hello word"); 
  12.  
  13. //get StringMsg 
  14. RxWebSocketUtil.getInstance().getWebSocketString(url) 
  15.         .subscribe(new Action1<String>() { 
  16.             @Override 
  17.             public void call(String s) { 
  18.             } 
  19.         }); 
  20. // get ByteString 
  21. RxWebSocketUtil.getInstance().getWebSocketByteString(url) 
  22.         .subscribe(new Action1<ByteString>() { 
  23.             @Override 
  24.             public void call(ByteString byteString) { 
  25.             } 
  26.         }); 
  27. //get WebSocket 
  28. RxWebSocketUtil.getInstance().getWebSocket(url) 
  29.         .subscribe(new Action1<WebSocket>() { 
  30.             @Override 
  31.             public void call(WebSocket webSocket) { 
  32.             } 
  33.         }); 
  34. // 帶timeout的WebSocket,當(dāng)在指定時(shí)間內(nèi)沒(méi)有收到消息,就重連WebSocket.為了適配小米平板. 
  35. RxWebSocketUtil.getInstance().getWebSocketInfo(url,10, TimeUnit.SECONDS) 
  36.         .subscribe(new Action1<WebSocketInfo>() { 
  37.             @Override 
  38.             public void call(WebSocketInfo webSocketInfo) { 
  39.             } 
  40.         }); 

2.發(fā)送消息:

  1. //用WebSocket的引用直接發(fā) 
  2. mWebSocket.send("hello word"); 
  3.  
  4. //url 對(duì)應(yīng)的WebSocket 必須打開(kāi),否則報(bào)錯(cuò) 
  5. RxWebSocket.send(sendUrl, "hello"); 
  6. RxWebSocket.send(sendUrl, ByteString.EMPTY); 
  7.  
  8. //異步發(fā)送,若WebSocket已經(jīng)打開(kāi),直接發(fā)送,若沒(méi)有打開(kāi),打開(kāi)一個(gè)WebSocket發(fā)送完數(shù)據(jù),直接關(guān)閉. 
  9. RxWebSocket.asyncSend(sendUrl, "hello"); 
  10. RxWebSocket.asyncSend(sendUrl, ByteString.EMPTY); 

3.關(guān)閉WebSocket:

項(xiàng)目是依托RxJava實(shí)現(xiàn)的,所以關(guān)閉WebSocket的方法也就是在適當(dāng)?shù)臅r(shí)候注銷(xiāo) Observable,項(xiàng)目里的demo里,寫(xiě)了一個(gè)簡(jiǎn)單的lifecycle,將Observable生命綁定到Activity的onDestroy,自動(dòng)注銷(xiāo).代碼細(xì)節(jié)請(qǐng)看demo,因?yàn)閮?nèi)部實(shí)現(xiàn)了同一個(gè)URL的WebSocket共享機(jī)制,所以當(dāng)外部所有持有這個(gè)URL的Observable都注銷(xiāo)后,這個(gè)WebSocket連接就會(huì)自動(dòng)關(guān)閉.請(qǐng)看原理解析部分

  1. //注意取消訂閱,有多種方式,比如 rxlifecycle 
  2. mSubscription = RxWebSocketUtil.getInstance().getWebSocketInfo(url) 
  3.         .subscribe(new Action1<WebSocketInfo>() { 
  4.             @Override 
  5.             public void call(WebSocketInfo webSocketInfo) { 
  6.                 mWebSocket = webSocketInfo.getWebSocket(); 
  7.  
  8.                 if (webSocketInfo.isOnOpen()) { 
  9.                     Log.d("MainActivity"" on WebSocket open"); 
  10.                 } else { 
  11.                     String string = webSocketInfo.getString(); 
  12.  
  13.                     if (string != null) { 
  14.                         Log.d("MainActivity", string); 
  15.                         textview.setText(Html.fromHtml(string)); 
  16.                     } 
  17.  
  18.                     ByteString byteString = webSocketInfo.getByteString(); 
  19.  
  20.                     if (byteString != null) { 
  21.                         Log.d("MainActivity"
  22.                             "webSocketInfo.getByteString():" + 
  23.                             byteString); 
  24.                     } 
  25.                 } 
  26.             } 
  27.         }); 
  28.  
  29. //注銷(xiāo) 
  30. if (mSubscription != null) { 
  31.     mSubscription.unsubscribe(); 
  32.  
  33. //lifecycle注銷(xiāo),詳情看demo 
  34. RxWebSocketUtil.getInstance().getWebSocketString(url) 
  35.             .compose(this.<String>bindOnActivityEvent(ActivityEvent.onDestory)) 
  36.             .subscribe(new Action1<String>() { 
  37.                 @Override 
  38.                 public void call(String s) { 
  39.                 } 
  40.             }); 

三. 原理解析

1. 首先需要將okhttp的WebSocket包裝成Observable,由于需要將WebSocket,Stringmsg,ByteString等信息一同發(fā)送給觀察者所以先構(gòu)建一個(gè)WebSocketInfo類(lèi),將信息封裝:

  1. public class WebSocketInfo { 
  2.     private WebSocket mWebSocket; 
  3.     private String mString; 
  4.     private ByteString mByteString; 
  5.     private boolean onOpen; 
  6.     //其他省略 

onOpen字段主要用來(lái)判斷當(dāng)前的這個(gè)WebSocketInfo是否是當(dāng)WebSocket打開(kāi)時(shí)發(fā)送的消息(onOpen),這時(shí),Stringmsg和ByteString都是null.

2. 將WebSocketInfo包裝成Observable發(fā)出:

  1. private final class WebSocketOnSubscribe implements Observable.OnSubscribe<WebSocketInfo> { 
  2.     private String url; 
  3.     private WebSocket webSocket; 
  4.     private WebSocketInfo startInfo; 
  5.     private WebSocketInfo stringInfo; 
  6.     private WebSocketInfo byteStringInfo; 
  7.  
  8.     public WebSocketOnSubscribe(String url) { 
  9.         this.url = url; 
  10.         startInfo = new WebSocketInfo(true); 
  11.         stringInfo = new WebSocketInfo(); 
  12.         byteStringInfo = new WebSocketInfo(); 
  13.     } 
  14.  
  15.     @Override 
  16.     public void call(final Subscriber<?super WebSocketInfo> subscriber) { 
  17.         if (webSocket != null) { 
  18.             //降低重連頻率 
  19.             if (!"main".equals(Thread.currentThread().getName())) { 
  20.                 SystemClock.sleep(2000); 
  21.             } 
  22.         } 
  23.  
  24.         initWebSocket(subscriber); 
  25.     } 
  26.  
  27.     private void initWebSocket( 
  28.         final Subscriber<?super WebSocketInfo> subscriber) { 
  29.         webSocket = client.newWebSocket(getRequest(url), 
  30.                 new WebSocketListener() { 
  31.                     @Override 
  32.                     public void onOpen(final WebSocket webSocket, 
  33.                         Response response) { 
  34.                         if (showLog) { 
  35.                             Log.d("RxWebSocketUtil", url + " --> onOpen"); 
  36.                         } 
  37.  
  38.                         webSocketMap.put(url, webSocket); 
  39.                         AndroidSchedulers.mainThread().createWorker().schedule(new Action0() { 
  40.                                 @Override 
  41.                                 public void call() { 
  42.                                     if (!subscriber.isUnsubscribed()) { 
  43.                                         subscriber.onStart(); 
  44.                                         startInfo.setWebSocket(webSocket); 
  45.                                         subscriber.onNext(startInfo); 
  46.                                     } 
  47.                                 } 
  48.                             }); 
  49.                     } 
  50.  
  51.                     @Override 
  52.                     public void onMessage(WebSocket webSocket, String text) { 
  53.                         if (!subscriber.isUnsubscribed()) { 
  54.                             stringInfo.setWebSocket(webSocket); 
  55.                             stringInfo.setString(text); 
  56.                             subscriber.onNext(stringInfo); 
  57.                         } 
  58.                     } 
  59.  
  60.                     @Override 
  61.                     public void onMessage(WebSocket webSocket, ByteString bytes) { 
  62.                         if (!subscriber.isUnsubscribed()) { 
  63.                             byteStringInfo.setWebSocket(webSocket); 
  64.                             byteStringInfo.setByteString(bytes); 
  65.                             subscriber.onNext(byteStringInfo); 
  66.                         } 
  67.                     } 
  68.  
  69.                     @Override 
  70.                     public void onFailure(WebSocket webSocket, Throwable t, 
  71.                         Response response) { 
  72.                         if (showLog) { 
  73.                             Log.e("RxWebSocketUtil"
  74.                                 t.toString() + 
  75.                                 webSocket.request().url().uri().getPath()); 
  76.                         } 
  77.  
  78.                         if (!subscriber.isUnsubscribed()) { 
  79.                             subscriber.onError(t); 
  80.                         } 
  81.                     } 
  82.  
  83.                     @Override 
  84.                     public void onClosing(WebSocket webSocket, int code, 
  85.                         String reason) { 
  86.                         webSocket.close(1000, null); 
  87.                     } 
  88.  
  89.                     @Override 
  90.                     public void onClosed(WebSocket webSocket, int code, 
  91.                         String reason) { 
  92.                         if (showLog) { 
  93.                             Log.d("RxWebSocketUtil"
  94.                                 url + " --> onClosed:code= " + code); 
  95.                         } 
  96.                     } 
  97.                 }); 
  98.         subscriber.add(new MainThreadSubscription() { 
  99.                 @Override 
  100.                 protected void onUnsubscribe() { 
  101.                     webSocket.close(3000, "手動(dòng)關(guān)閉"); 
  102.                 } 
  103.             }); 
  104.     } 

實(shí)現(xiàn)一個(gè)WebSocketOnSubscribe 將WebSocket的回調(diào)轉(zhuǎn)化成subscriber調(diào)用.發(fā)送給Observable下游.在onOpen時(shí)調(diào)用 subscriber.onStart(),并且發(fā)送一個(gè)onOpen的WebSocketInfo.在subscriber注銷(xiāo)的時(shí)候關(guān)閉WebSocket.在call方法最上面有個(gè)SystemClock.sleep(2000),這個(gè)主要是為了降低在斷連的時(shí)候的重連頻率,將在下面講到.

包裝成Observable:

  1. Observable.create(new WebSocketOnSubscribe(url)) 
  2.             .subscribeOn(Schedulers.io()) 
  3.             .observeOn(AndroidSchedulers.mainThread()); 

3. 實(shí)現(xiàn)自動(dòng)重連:

  1. Observable.create(new WebSocketOnSubscribe(url)) 
  2.             //自動(dòng)重連 
  3.             .timeout(timeout, timeUnit).retry() 
  4.             .subscribeOn(Schedulers.io()) 
  5.             .observeOn(AndroidSchedulers.mainThread()); 

RxJava retry操作符,很完美的實(shí)現(xiàn)了這個(gè)功能,當(dāng)上游發(fā)出Throwable的時(shí)候,retry將錯(cuò)誤吃掉,并重新調(diào)用 onSubscribe的call方法,也就是WebSocketOnSubscribe的call,就會(huì)重新初始化一個(gè)WebSocket連接,達(dá)到重連的目的,如果一直沒(méi)有網(wǎng)絡(luò),這個(gè)retry的調(diào)用頻率非常高,所以在call方法里面,當(dāng)是重連的時(shí)候,就SystemClock.sleep(2000),休眠2秒,這樣重連的頻率就是2秒重連一次. 當(dāng)然在retry上面還有一個(gè)timeout操作符.當(dāng)subscriber.onNext()在指定時(shí)間間隔里沒(méi)有調(diào)用,就發(fā)出一個(gè)timeoutException,讓retry重連WebSocket.這個(gè)主要是為了適配部分國(guó)產(chǎn)機(jī)型,當(dāng)WebSocket發(fā)生連接異常時(shí),不會(huì)及時(shí)發(fā)出錯(cuò)誤,如小米平板.在每次重連都會(huì)把原來(lái)的WebSocket關(guān)閉.

4. 實(shí)現(xiàn)同一個(gè)URL的WebSocket共享

  1. Observable.create(new WebSocketOnSubscribe(url)) 
  2.         //自動(dòng)重連 
  3.         .timeout(timeout, timeUnit) 
  4.         .retry() 
  5.         //共享 
  6.         .doOnUnsubscribe(new Action0() { 
  7.             @Override 
  8.             public void call() { 
  9.                 observableMap.remove(url); 
  10.                 webSocketMap.remove(url); 
  11.  
  12.                 if (showLog) { 
  13.                     Log.d("RxWebSocketUtil""注銷(xiāo)"); 
  14.                 } 
  15.             } 
  16.         }) 
  17.         .doOnNext(new Action1<WebSocketInfo>() { 
  18.             @Override 
  19.             public void call(WebSocketInfo webSocketInfo) { 
  20.                 if (webSocketInfo.isOnOpen()) { 
  21.                     webSocketMap.put(url, webSocketInfo.getWebSocket()); 
  22.                 } 
  23.             } 
  24.         }) 
  25.         .share() 
  26.         .subscribeOn(Schedulers.io()) 
  27.         .observeOn(AndroidSchedulers.mainThread()); 

實(shí)現(xiàn)共享功能,主要是為了防止一個(gè)URL的WebSocket,建立多個(gè)連接,這個(gè)主要是由RxJava的share操作符實(shí)現(xiàn),share操作符,使得一個(gè)Observable可以有多個(gè)subscriber,當(dāng)有多個(gè)subscriber時(shí),當(dāng)所有的subscriber都取消訂閱,這個(gè)Observable才會(huì)取消訂閱.

getWebSocketInfo()方法完整代碼:

  1. public Observable<WebSocketInfo> getWebSocketInfo(final String url, final long timeout, final TimeUnit timeUnit) { 
  2.     Observable<WebSocketInfo> observable = observableMap.get(url); 
  3.     if (observable == null) { 
  4.         observable = Observable.create(new WebSocketOnSubscribe(url)) 
  5.                 //自動(dòng)重連 
  6.                 .timeout(timeout, timeUnit) 
  7.                 .retry() 
  8.                 //共享 
  9.                 .doOnUnsubscribe(new Action0() { 
  10.                     @Override 
  11.                     public void call() { 
  12.                         observableMap.remove(url); 
  13.                         webSocketMap.remove(url); 
  14.                         if (showLog) { 
  15.                             Log.d("RxWebSocketUtil""注銷(xiāo)"); 
  16.                         } 
  17.                     } 
  18.                 }) 
  19.                 .doOnNext(new Action1<WebSocketInfo>() { 
  20.                     @Override 
  21.                     public void call(WebSocketInfo webSocketInfo) { 
  22.                         if (webSocketInfo.isOnOpen()) { 
  23.                             webSocketMap.put(url, webSocketInfo.getWebSocket()); 
  24.                         } 
  25.                     } 
  26.                 }) 
  27.                 .share() 
  28.                 .subscribeOn(Schedulers.io()) 
  29.                 .observeOn(AndroidSchedulers.mainThread()); 
  30.         observableMap.put(url, observable); 
  31.     } else { 
  32.         observable = Observable.merge(Observable.just(new WebSocketInfo(webSocketMap.get(url), true)), observable); 
  33.     } 
  34.     return observable; 

doOnUnsubscribe作用:在Observable注銷(xiāo),即 WebSocket關(guān)閉時(shí),移除map中的緩存的Observable和WebSocket.

doOnNext作用: 判斷接收到的WebSocketInfo是否是WebSocket在onOpen的時(shí)候發(fā)的,然后將其緩存起來(lái).作用就是:如果有一個(gè)相同的URL訂閱Observable,就從緩存中取,這個(gè)時(shí)候我們應(yīng)該把一個(gè)WebSocket的onOpen事件也發(fā)給這個(gè)訂閱者:

  1. //使用merge操作符,將onOpen事件發(fā)給訂閱者 
  2. observable = Observable.merge(Observable.just(new WebSocketInfo(webSocketMap.get(url), true)), observable); 

這樣的話,同一個(gè)URL的WebSocket,不管在什么地方什么時(shí)間訂閱,都能收到一個(gè)onOpen事件,外部表現(xiàn)的就像一個(gè)新的WebSocket.

getWebSocketInfo方法的幾種變體:

  1. /** 
  2.  * default timeout: 30 days 
  3.  * <p> 
  4.  * 若忽略小米平板,請(qǐng)調(diào)用這個(gè)方法 
  5.  * </p> 
  6.  */ 
  7. public Observable<WebSocketInfo> getWebSocketInfo(String url) { 
  8.     return getWebSocketInfo(url, 30, TimeUnit.DAYS); 
  9.  
  10. public Observable<String> getWebSocketString(String url) { 
  11.     return getWebSocketInfo(url) 
  12.             .map(new Func1<WebSocketInfo, String>() { 
  13.                 @Override 
  14.                 public String call(WebSocketInfo webSocketInfo) { 
  15.                     return webSocketInfo.getString(); 
  16.                 } 
  17.             }) 
  18.             .filter(new Func1<String, Boolean>() { 
  19.                 @Override 
  20.                 public Boolean call(String s) { 
  21.                     return s != null
  22.                 } 
  23.             }); 
  24.  
  25. public Observable<ByteString> getWebSocketByteString(String url) { 
  26.     return getWebSocketInfo(url) 
  27.             .map(new Func1<WebSocketInfo, ByteString>() { 
  28.                 @Override 
  29.                 public ByteString call(WebSocketInfo webSocketInfo) { 
  30.                     return webSocketInfo.getByteString(); 
  31.                 } 
  32.             }) 
  33.             .filter(new Func1<ByteString, Boolean>() { 
  34.                 @Override 
  35.                 public Boolean call(ByteString byteString) { 
  36.                     return byteString != null
  37.                 } 
  38.             }); 
  39.  
  40. public Observable<WebSocket> getWebSocket(String url) { 
  41.     return getWebSocketInfo(url) 
  42.             .map(new Func1<WebSocketInfo, WebSocket>() { 
  43.                 @Override 
  44.                 public WebSocket call(WebSocketInfo webSocketInfo) { 
  45.                     return webSocketInfo.getWebSocket(); 
  46.                 } 
  47.             }); 

5 . send信息到服務(wù)端

上面已經(jīng)講到WebSocketInfo包含了WebSocket,所以在訂閱后,就可以拿到這個(gè)WebSocket引用就可以WebSocket.send發(fā)送消息到服務(wù)端.當(dāng)然我們的RxWebSocketUtil已經(jīng)將開(kāi)啟的WebSocket已經(jīng)緩存.所以我們也可以這樣發(fā)消息:

  1. /** 
  2.  * 如果url的WebSocket已經(jīng)打開(kāi),可以直接調(diào)用這個(gè)發(fā)送消息. 
  3.  * 
  4.  * @param url 
  5.  * @param msg 
  6.  */ 
  7. public void send(String url, String msg) { 
  8.     WebSocket webSocket = webSocketMap.get(url); 
  9.     if (webSocket != null) { 
  10.         webSocket.send(msg); 
  11.     } else { 
  12.         throw new IllegalStateException("The WebSokcet not open"); 
  13.     } 
  14.  
  15. /** 
  16.  * 如果url的WebSocket已經(jīng)打開(kāi),可以直接調(diào)用這個(gè)發(fā)送消息. 
  17.  * 
  18.  * @param url 
  19.  * @param byteString 
  20.  */ 
  21. public void send(String url, ByteString byteString) { 
  22.     WebSocket webSocket = webSocketMap.get(url); 
  23.     if (webSocket != null) { 
  24.         webSocket.send(byteString); 
  25.     } else { 
  26.         throw new IllegalStateException("The WebSokcet not open"); 
  27.     } 

當(dāng)指定的URL的WebSocket沒(méi)有打開(kāi)會(huì)直接報(bào)錯(cuò).

異步發(fā)送消息到服務(wù)端

  1. /** 
  2.  * 不用關(guān)心url 的WebSocket是否打開(kāi),可以直接發(fā)送 
  3.  * 
  4.  * @param url 
  5.  * @param msg 
  6.  */ 
  7. public void asyncSend(String url, final String msg) { 
  8.     getWebSocket(url) 
  9.             .first() 
  10.             .subscribe(new Action1<WebSocket>() { 
  11.                 @Override 
  12.                 public void call(WebSocket webSocket) { 
  13.                     webSocket.send(msg); 
  14.                 } 
  15.             }); 
  16.  
  17.  
  18. /** 
  19.  * 不用關(guān)心url 的WebSocket是否打開(kāi),可以直接發(fā)送 
  20.  * 
  21.  * @param url 
  22.  * @param byteString 
  23.  */ 
  24. public void asyncSend(String url, final ByteString byteString) { 
  25.     getWebSocket(url) 
  26.             .first() 
  27.             .subscribe(new Action1<WebSocket>() { 
  28.                 @Override 
  29.                 public void call(WebSocket webSocket) { 
  30.                     webSocket.send(byteString); 
  31.                 } 
  32.             }); 

這兩種發(fā)送方式,你不用關(guān)心URL的WebSocket是否打開(kāi),可以直接發(fā)送.實(shí)現(xiàn)思路也很簡(jiǎn)單,getWebSocket(url)會(huì)獲取到Observable,或者是從緩存中取,或者是重新開(kāi)啟一個(gè)WebSocket,但你都不需要關(guān)心,經(jīng)過(guò)first操作符后,如果是從緩存取的Observable,就注銷(xiāo)的當(dāng)前的Observable,當(dāng)是新開(kāi)的WebSocket,注銷(xiāo)掉當(dāng)前的subscriber后,就沒(méi)有其他subscriber了,這個(gè)新開(kāi)的WebSocket就會(huì)關(guān)閉(share操作符作用).

想了解更多內(nèi)容,請(qǐng)?jiān)L問(wèn):

51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)

https://harmonyos.51cto.com

 

責(zé)任編輯:jianghua 來(lái)源: 鴻蒙社區(qū)
相關(guān)推薦

2023-12-11 07:12:21

心跳檢測(cè)重連機(jī)制服務(wù)端

2021-02-26 12:37:39

WebSocketOkHttp連接

2017-05-25 11:49:30

Android網(wǎng)絡(luò)請(qǐng)求OkHttp

2025-08-19 17:32:16

2024-03-19 08:45:45

WebSocketSpring應(yīng)用開(kāi)發(fā)

2021-12-07 10:23:27

鴻蒙HarmonyOS應(yīng)用

2021-11-09 09:30:52

OkHttp面試Android

2014-09-01 10:22:11

Node.js技術(shù)架構(gòu)

2020-03-19 10:13:13

OkHttpWebSocket

2017-03-22 13:20:07

RxJavaSingleCompletable

2024-04-03 15:40:14

WebSocketWeb應(yīng)用Spring

2018-04-20 09:36:23

NettyWebSocket京東

2024-11-04 08:00:00

Netty客戶端

2024-12-23 06:00:00

TCPC#網(wǎng)絡(luò)

2024-08-07 08:22:27

2012-06-04 14:41:16

Win7連網(wǎng)

2014-03-25 14:21:18

WebSocket實(shí)時(shí)

2025-09-12 00:00:01

HTTP服務(wù)器Spring AI

2024-01-26 08:41:55

Fluent工具包高并發(fā)

2025-06-20 02:11:00

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

精品人妻一区二区三区香蕉 | 国产精品伦理一区| 香蕉久久网站| 亚洲国产精品va| 天堂在线资源视频| 男人添女人下部高潮视频在线观看 | 亚洲卡通欧美制服中文| 久久99精品久久久久子伦| 中文人妻熟女乱又乱精品| 国产精品sm| 国产亚洲激情视频在线| 91porn在线| 国产精品久久久久77777丨| 亚洲一区中文在线| 一本色道久久综合亚洲二区三区| 人妻少妇一区二区三区| 极品少妇xxxx精品少妇偷拍| 97超视频免费观看| 国产一区二区播放| 精品理论电影在线| 日韩国产在线播放| 农村末发育av片一区二区| 国产激情久久| 在线精品视频免费观看| 国产a级片网站| av网站网址在线观看| 久久久综合精品| 国产激情美女久久久久久吹潮| 伊人网av在线| 日韩不卡免费视频| 91成人福利在线| 免费在线观看国产精品| 亚洲精品国产偷自在线观看| 日韩在线观看网站| 中文字幕伦理片| 久草成人资源| 亚洲女同精品视频| 精品人妻一区二区三区视频| swag国产精品一区二区| 欧美一个色资源| 国产美女视频免费看| 精品日本视频| 91福利在线观看| av免费中文字幕| 中文在线аv在线| 欧美日韩激情小视频| cao在线观看| 毛片网站在线看| 亚洲电影一区二区| 波多野结衣av一区二区全免费观看| 老司机99精品99| 中文字幕一区二区三区不卡在线| 亚洲.欧美.日本.国产综合在线| 你懂的在线视频| 久久久久久久久一| 日韩欧美亚洲日产国| 黄色片在线看| 国产精品亲子乱子伦xxxx裸| 亚洲图片欧洲图片日韩av| 欧美捆绑视频| 中文字幕不卡的av| 在线亚洲美日韩| 26uuu亚洲电影在线观看| 亚洲精品免费在线| 青青草成人免费在线视频| 日本不卡网站| 在线看日本不卡| 色戒在线免费观看| 视频一区视频二区欧美| 亚洲电影免费观看高清完整版在线观看| 激情综合激情五月| 天天躁日日躁成人字幕aⅴ| 亚洲色图激情小说| 久久嫩草捆绑紧缚| 欧美午夜在线| 日本欧美国产在线| 国产又黄又粗又长| 成人h精品动漫一区二区三区| 久久久人人爽| 午夜激情视频在线观看| 亚洲狠狠丁香婷婷综合久久久| 岛国大片在线播放| 日韩久久一区二区三区| 欧美久久一区二区| 老司机午夜免费福利| 久久99性xxx老妇胖精品| 久久久精品久久久| 男人的天堂一区二区| 日本不卡免费在线视频| 亚洲综合色av| 国产午夜在线观看| 一区二区三区小说| 成人免费xxxxx在线视频| 国产精品美女久久久久| 亚洲精品一区二区久| 久久久久久久麻豆| 亚洲欧美日韩国产综合精品二区| 国产欧美va欧美va香蕉在线| 欧美特级特黄aaaaaa在线看| 中文乱码免费一区二区| 69sex久久精品国产麻豆| 精品123区| 日韩电影在线观看永久视频免费网站| 五月天婷婷丁香网| 免费永久网站黄欧美| 亚洲www在线| 久久伊伊香蕉| 亚洲www啪成人一区二区麻豆| 日本 片 成人 在线| 欧美aaaaa级| 成年人精品视频| 国产免费www| 国产精品久久久久久| 性欧美videos高清hd4k| 午夜精品久久久久久久99水蜜桃| 国产一区二区在线免费播放| 看全色黄大色大片免费久久久| 色综合伊人色综合网| 秋霞精品一区二区三区| 成人精品一区二区三区中文字幕| 日韩欧美一区二区三区久久婷婷| 欧洲一区二区三区| 在线播放91灌醉迷j高跟美女| 一本色道久久综合亚洲精品图片 | 日本一区免费在线观看| 金瓶狂野欧美性猛交xxxx| 欧美精品久久99| www.涩涩爱| 日本女人一区二区三区| 日本不卡一区二区三区在线观看| 丁香花在线观看完整版电影| 日韩视频一区二区三区在线播放| 强制高潮抽搐sm调教高h| 蜜臀久久99精品久久久久久9| 欧美精品亚洲| 中文日产幕无线码一区二区| 亚洲精品福利在线观看| 国产午夜精品无码| 成人精品国产免费网站| 国产精品久久久久久久久电影网| 国产精品亚洲综合在线观看| 日韩在线播放视频| 一区二区日韩视频| 成人免费在线播放视频| 久久成年人网站| 牛牛国产精品| 国产98在线|日韩| 波多野结衣中文字幕久久| 欧美第一区第二区| 国产精品美女毛片真酒店| youjizz久久| 日韩免费一级视频| 久久综合亚洲| 国产在线精品自拍| www视频在线免费观看| 日韩欧美成人激情| 国产特黄大片aaaa毛片| 91视频精品在这里| 天堂社区在线视频| 91欧美在线| 99在线观看视频| av午夜在线观看| 亚洲精品在线观看www| 激情网站在线观看| 中文字幕一区二区在线观看| 毛毛毛毛毛毛毛片123| 精品成人国产| 欧洲精品码一区二区三区免费看| 成人午夜一级| 欧美理论电影在线观看| 色鬼7777久久| 欧美视频精品在线观看| 免费在线黄色片| 91蜜桃在线观看| 久久撸在线视频| 欧美午夜一区| 欧美一区二区三区精美影视 | 97影院秋霞午夜在线观看| 欧美不卡一二三| 亚洲毛片一区二区三区| 亚洲欧洲日韩综合一区二区| 老熟女高潮一区二区三区| 久久国产毛片| 免费看污污视频| 五月天亚洲色图| 亚洲一区二区自拍| 中文字幕一区久| 欧美美女15p| 国产高清免费在线播放| 日韩欧美国产wwwww| 久久久久久无码精品大片| 亚洲品质自拍视频网站| 亚洲专区区免费| 国产一区二区精品在线观看| 国产在线青青草| 欧美1区3d| 先锋影音亚洲资源| 国产色噜噜噜91在线精品 | 国产精品女人久久久| 天天操天天干天天综合网| sm捆绑调教视频| 久久一区二区三区四区| 欧美激情一区二区三区p站| 麻豆国产精品官网| 日本wwww视频| 欧美激情第10页| 自拍另类欧美| 国内黄色精品| 精品视频一区在线| 一区二区在线免费播放| 国产精品爽黄69天堂a| 蜜桃视频在线观看免费视频| 欧美xxxx18国产| 日本韩国在线视频爽| 亚洲视频欧洲视频| 香港一级纯黄大片| 精品国精品国产| 国产免费一区二区三区免费视频| 在线观看91视频| 中文字幕一区在线播放| 午夜精品福利一区二区蜜股av| 好吊日在线视频| 中文字幕一区二区不卡| 婷婷色一区二区三区| 97国产一区二区| 国产午夜在线一区二区三区| 国产成人在线视频网址| 欧美激情国内自拍| 韩国女主播成人在线| 国产免费又粗又猛又爽| 三级一区在线视频先锋| 免费日韩视频在线观看| 国产欧美丝祙| 欧美日韩成人免费视频| 亚洲一区二区三区免费在线观看| 丁香六月激情婷婷| 欧美fxxxxxx另类| av日韩在线看| 尹人成人综合网| 免费看国产曰批40分钟| 国产欧美欧美| 一本大道熟女人妻中文字幕在线| 亚洲日产国产精品| 精品久久久久久久久久中文字幕| 国一区二区在线观看| 成人免费在线网| 国产一区二区三区成人欧美日韩在线观看 | 日本少妇一级片| 国产91对白在线观看九色| 精品伦一区二区三区| 成人美女视频在线观看18| 182在线视频| 久久免费看少妇高潮| 国产jjizz一区二区三区视频| 中文字幕欧美激情一区| 91导航在线观看| 亚洲免费观看高清在线观看| 久久中文字幕在线观看| 欧美午夜丰满在线18影院| 国产99免费视频| 在线不卡中文字幕| 亚洲欧美另类视频| 日韩www在线| 国产在线免费观看| 精品国偷自产在线视频| 美女精品导航| 日本久久久a级免费| 欧洲成人一区| 97av自拍| 亚洲免费专区| 精品一区二区成人免费视频| 国产精品magnet| 蜜臀久久99精品久久久酒店新书 | 国产成人精品免费视频大全最热| 精品亚洲免a| 天天好比中文综合网| 亚洲欧美日韩高清在线| 国产精品网站免费| 免费在线成人网| 中文字幕人妻一区| 国产精品色在线观看| 精品爆乳一区二区三区无码av| 懂色aⅴ精品一区二区三区蜜月| 四虎影院在线免费播放| 日韩三级视频在线观看| 男女av在线| 欧美激情2020午夜免费观看| 3d性欧美动漫精品xxxx软件| 91成人在线看| 精品国产乱码久久久| 日本久久久网站| 久久精品国产999大香线蕉| 国产精品久久久久久亚洲色| 中文字幕精品一区二区三区精品| 国产无遮挡aaa片爽爽| 欧美日韩精品一区二区三区| 午夜av免费观看| 欧美精品免费看| 成人在线高清| 免费一区二区三区| 欧美特黄一区| 国产亚洲视频一区| 国产蜜臀97一区二区三区| 国产精品99精品| 欧美一区二区在线免费播放| 免费在线看v| 91av在线免费观看视频| 欧美日韩国产一区二区在线观看| 日韩videos| 亚洲欧美视频| 精品视频站长推荐| 一区二区三区毛片| 国产一区二区三区四区视频| 亚洲人精品午夜在线观看| a天堂资源在线| 91视频在线免费观看| 欧美aaaa视频| 美女网站视频黄色| 久久奇米777| 97人人澡人人爽人人模亚洲| 欧美一区二区三区视频在线观看 | 久久久久这里只有精品| 亚洲欧洲一二区| 先锋影音亚洲资源| 青草av.久久免费一区| www在线观看免费视频| 欧美日韩在线影院| 天堂中文在线观看视频| 久久免费视频在线| 91成人午夜| 国产中文字幕乱人伦在线观看| 国产精品69久久久久水密桃 | 欧美性黄网官网| 水莓100国产免费av在线播放| 91高清视频在线免费观看| 大陆精大陆国产国语精品| 日韩在线视频在线| 国产成人午夜99999| 青青草手机在线视频| 欧美一区二区日韩一区二区| 麻豆视频在线观看免费网站| 成人福利网站在线观看| 久久久久久久久久久久久久| 亚洲综合激情视频| 亚洲少妇30p| 99国产精品久久久久久久成人| 精品国产一区二区三区久久狼黑人 | 久久久久久久久久久国产精品| 精品久久久一区| 美女做暖暖视频免费在线观看全部网址91 | 久久精品美女视频| 精品处破学生在线二十三| 99热99re6国产在线播放| 精品久久久久久一区| 校园激情久久| 美国美女黄色片| 91麻豆精品国产91久久久久久久久| av中文字幕在线播放| 国产精品一区二区三区观看| 99国产精品视频免费观看一公开 | 欧美日韩一区在线观看| 久操视频在线免费播放| 国产精品国产三级国产专区53 | 三级视频网站在线观看| 欧美日韩亚洲一区二| 91porn在线观看| 99三级在线| 久久综合九色综合欧美狠狠| 亚洲无人区码一码二码三码的含义| 欧美人与禽zozo性伦| 国产网红在线观看| 日本在线播放不卡| 激情文学综合插| 国产精品午夜影院| 色香阁99久久精品久久久| 999在线精品| 亚洲爆乳无码专区| 亚洲免费伊人电影| 亚州精品国产精品乱码不99按摩| 国产精品久久999| 国产精品vip| 法国空姐电影在线观看| 91精品国产综合久久久蜜臀粉嫩 | 国产精品豆花视频| 精品无码在线观看| 精品久久久久久久久久久久久久久久久| 在线中文字幕播放| 国产手机视频在线观看| 26uuu久久天堂性欧美| av男人天堂网| 国产精品久久久久久久久久新婚| 午夜精品亚洲| 亚洲色图欧美色| 亚洲精品电影在线| 国产精品久久免费视频| 欧美激情精品久久久久久小说| 亚洲码国产岛国毛片在线| 国产主播福利在线| 国产精品夜夜夜一区二区三区尤| 麻豆精品在线播放|