日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

OkHttp实现分析之Websocket

發(fā)布時(shí)間:2024/4/11 编程问答 57 豆豆
生活随笔 收集整理的這篇文章主要介紹了 OkHttp实现分析之Websocket 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

HTML5 擁有許多引人注目的新特性,WebSocket就是其中之一。WebSocket一向有著 “Web 的 TCP ”之稱。通常 WebSocket 都是用于Web的,用于構(gòu)建實(shí)時(shí)的 Web 應(yīng)用。它可以在瀏覽器和服務(wù)器之間提供一個(gè)基于 TCP 連接的雙向通道。

WebSocket 協(xié)議本質(zhì)上是一個(gè)基于 TCP 的協(xié)議。為了建立一個(gè) WebSocket 連接,客戶端瀏覽器首先要向服務(wù)器發(fā)起一個(gè) HTTP 請(qǐng)求,這個(gè)請(qǐng)求和通常的 HTTP 請(qǐng)求不同,包含了一些附加頭信息,其中附加頭信息 ”Upgrade: WebSocket” 表明這是一個(gè)申請(qǐng)協(xié)議升級(jí)的 HTTP 請(qǐng)求,服務(wù)器端解析這些附加的頭信息然后產(chǎn)生應(yīng)答信息返回給客戶端,客戶端和服務(wù)器端的 WebSocket 連接就建立起來(lái)了,雙方就可以通過(guò)這個(gè)連接通道自由的傳遞信息,并且這個(gè)連接會(huì)持續(xù)存在直到客戶端或者服務(wù)器端的某一方主動(dòng)的關(guān)閉連接。

Websocket同樣可以用于移動(dòng)端。盡管移動(dòng)端 Android/iOS 的本地應(yīng)用可以直接通過(guò)Socket與服務(wù)器建立連接,并定義自己的協(xié)議來(lái)解決 Web 中實(shí)時(shí)應(yīng)用創(chuàng)建困難的問(wèn)題,但 WebSocket 服務(wù)通常復(fù)用Web的 80 端口,且可以比較方便的基于Web服務(wù)器來(lái)實(shí)現(xiàn),因而對(duì)于某些端口容易被封的網(wǎng)絡(luò)環(huán)境而言,WebSocket 就變得非常有意義。

OkHttp 是在 2016 年 6 月 10 日發(fā)布的 3.4.1 版中添加的對(duì)WebSocket的支持的。本文通過(guò)分析 OkHttp-3.5.0 的 WebSocket 實(shí)現(xiàn)來(lái)學(xué)習(xí)一下這個(gè)協(xié)議。

OkHttp WebSocket客戶端 API 用法

在開(kāi)始分析 WebSocket 的實(shí)現(xiàn)之前,我們先來(lái)看一下 OkHttp 的 WebSocket API怎么用。示例代碼如下:

import android.util.Log;import okhttp3.OkHttpClient; import okhttp3.Request; import okhttp3.Response; import okhttp3.WebSocket; import okhttp3.WebSocketListener; import okio.ByteString;public class WebsocketClient {private static final int NORMAL_CLOSURE_STATUS = 1000;private static OkHttpClient sClient;private static WebSocket sWebSocket;public static synchronized void startRequest() {if (sClient == null) {sClient = new OkHttpClient();}if (sWebSocket == null) {Request request = new Request.Builder().url("ws://echo.websocket.org").build();EchoWebSocketListener listener = new EchoWebSocketListener();sWebSocket = sClient.newWebSocket(request, listener);}}private static void sendMessage(WebSocket webSocket) {webSocket.send("Knock, knock!");webSocket.send("Hello!");webSocket.send(ByteString.decodeHex("deadbeef"));}public static void sendMessage() {WebSocket webSocket;synchronized (WebsocketClient.class) {webSocket = sWebSocket;}if (webSocket != null) {sendMessage(webSocket);}}public static synchronized void closeWebSocket() {if (sWebSocket != null) {sWebSocket.close(NORMAL_CLOSURE_STATUS, "Goodbye!");sWebSocket = null;}}public static synchronized void destroy() {if (sClient != null) {sClient.dispatcher().executorService().shutdown();sClient = null;}}private static void resetWebSocket() {synchronized (WebsocketClient.class) {sWebSocket = null;}}public static class EchoWebSocketListener extends WebSocketListener {private static final String TAG = "EchoWebSocketListener";@Overridepublic void onOpen(WebSocket webSocket, Response response) {sendMessage(webSocket);}@Overridepublic void onMessage(WebSocket webSocket, String text) {Log.i(TAG, "Receiving: " + text);}@Overridepublic void onMessage(WebSocket webSocket, ByteString bytes) {Log.i(TAG, "Receiving: " + bytes.hex());}@Overridepublic void onClosing(WebSocket webSocket, int code, String reason) {webSocket.close(NORMAL_CLOSURE_STATUS, null);Log.i(TAG, "Closing: " + code + " " + reason);resetWebSocket();}@Overridepublic void onClosed(WebSocket webSocket, int code, String reason) {Log.i(TAG, "Closed: " + code + " " + reason);}@Overridepublic void onFailure(WebSocket webSocket, Throwable t, Response response) {t.printStackTrace();resetWebSocket();}} }

這個(gè)過(guò)程與發(fā)送HTTP請(qǐng)求的過(guò)程有許多相似之處,它們都需要?jiǎng)?chuàng)建 OkHttpClient 和Request。然而它們不同的地方更多:

  • WebSocket 請(qǐng)求通過(guò) WebSocketListener 來(lái)接收連接的狀態(tài)和活動(dòng),而HTTP請(qǐng)求則通過(guò) Callback。同時(shí)請(qǐng)求的 URL 的 scheme 是 “ws” 或者是 “wss” (TLS 之上的 WebSocket),而不是HTTP的 "http" 和 "https"。
  • HTTP 請(qǐng)求的連接建立及執(zhí)行需要基于 Request 和回調(diào)創(chuàng)建Call,并調(diào)用 Call 的方法手動(dòng)進(jìn)行;而對(duì)于 WebSocket 請(qǐng)求,則在基于 Request 和回調(diào)創(chuàng)建 WebSocket 的時(shí)候,OkHttp 會(huì)自動(dòng)發(fā)起連接建立的過(guò)程。
  • 這也是 WebSocket 與 HTTP 最大的不同。對(duì)于 WebSocket,我們可以保存 WebSocket 對(duì)象,并在后續(xù)多次通過(guò)該對(duì)象向服務(wù)器發(fā)送數(shù)據(jù)。
  • 通過(guò)回調(diào)可以獲得更多 WebSocket 的狀態(tài)變化。在連接建立、收到服務(wù)器發(fā)送回來(lái)的消息、服務(wù)器要關(guān)閉連接,以及出現(xiàn) error 時(shí),都能得到通知。不像 HTTP 請(qǐng)求那樣,只在最后得到一個(gè)請(qǐng)求成功或者失敗的結(jié)果。
  • 后兩點(diǎn)正是 WebSocket 全雙工連接的體現(xiàn)。

    OkHttp 的 WebSocket 實(shí)現(xiàn)

    接著我們來(lái)看OkHttp 的 WebSocket 實(shí)現(xiàn)。WebSocket 包含兩個(gè)部分,分別是握手和數(shù)據(jù)傳輸,數(shù)據(jù)傳輸又包括數(shù)據(jù)的發(fā)送,數(shù)據(jù)的接收,連接的保活,以及連接的關(guān)閉等,我們將分別分析這些過(guò)程。

    連接握手

    創(chuàng)建 WebSocket 的過(guò)程如下:

    public class OkHttpClient implements Cloneable, Call.Factory, WebSocket.Factory {. . . . . .@Override public WebSocket newWebSocket(Request request, WebSocketListener listener) {RealWebSocket webSocket = new RealWebSocket(request, listener, new SecureRandom());webSocket.connect(this);return webSocket;}

    在這里會(huì)創(chuàng)建一個(gè) RealWebSocket 對(duì)象,然后執(zhí)行其 connect() 方法建立連接。 RealWebSocket 對(duì)象的創(chuàng)建過(guò)程如下:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .public RealWebSocket(Request request, WebSocketListener listener, Random random) {if (!"GET".equals(request.method())) {throw new IllegalArgumentException("Request must be GET: " + request.method());}this.originalRequest = request;this.listener = listener;this.random = random;byte[] nonce = new byte[16];random.nextBytes(nonce);this.key = ByteString.of(nonce).base64();this.writerRunnable = new Runnable() {@Override public void run() {try {while (writeOneFrame()) {}} catch (IOException e) {failWebSocket(e, null);}}};}

    這里最主要的是初始化了 key,以備后續(xù)連接建立及握手之用。Key 是一個(gè)16字節(jié)長(zhǎng)的隨機(jī)數(shù)經(jīng)過(guò) Base64 編碼得到的。此外還初始化了 writerRunnable 等。

    連接建立及握手過(guò)程如下:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .public void connect(OkHttpClient client) {client = client.newBuilder().protocols(ONLY_HTTP1).build();final int pingIntervalMillis = client.pingIntervalMillis();final Request request = originalRequest.newBuilder().header("Upgrade", "websocket").header("Connection", "Upgrade").header("Sec-WebSocket-Key", key).header("Sec-WebSocket-Version", "13").build();call = Internal.instance.newWebSocketCall(client, request);call.enqueue(new Callback() {@Override public void onResponse(Call call, Response response) {try {checkResponse(response);} catch (ProtocolException e) {failWebSocket(e, response);closeQuietly(response);return;}// Promote the HTTP streams into web socket streams.StreamAllocation streamAllocation = Internal.instance.streamAllocation(call);streamAllocation.noNewStreams(); // Prevent connection pooling!Streams streams = streamAllocation.connection().newWebSocketStreams(streamAllocation);// Process all web socket messages.try {listener.onOpen(RealWebSocket.this, response);String name = "OkHttp WebSocket " + request.url().redact();initReaderAndWriter(name, pingIntervalMillis, streams);streamAllocation.connection().socket().setSoTimeout(0);loopReader();} catch (Exception e) {failWebSocket(e, null);}}@Override public void onFailure(Call call, IOException e) {failWebSocket(e, null);}});}

    連接建立及握手的過(guò)程主要是向服務(wù)器發(fā)送一個(gè)HTTP請(qǐng)求。這個(gè) HTTP 請(qǐng)求的特別之處在于,它包含了如下的一些Headers:

    Upgrade: WebSocket Connection: Upgrade Sec-WebSocket-Key: 7wgaspE0Tl7/66o4Dov2kw== Sec-WebSocket-Version: 13

    其中 Upgrade 和 Connection header 向服務(wù)器表明,請(qǐng)求的目的就是要將客戶端和服務(wù)器端的通訊協(xié)議從 HTTP 協(xié)議升級(jí)到 WebSocket 協(xié)議,同時(shí)在請(qǐng)求處理完成之后,連接不要斷開(kāi)。Sec-WebSocket-Key header 值正是我們前面看到的key,它是 WebSocket 客戶端發(fā)送的一個(gè) base64 編碼的密文,要求服務(wù)端必須返回一個(gè)對(duì)應(yīng)加密的 “Sec-WebSocket-Accept” 應(yīng)答,否則客戶端會(huì)拋出 “Error during WebSocket handshake” 錯(cuò)誤,并關(guān)閉連接。

    來(lái)自于 HTTP 服務(wù)器的響應(yīng)到達(dá)的時(shí)候,即是連接建立大功告成的時(shí)候,也就是熱豆腐孰了的時(shí)候。

    然而,響應(yīng)到達(dá)時(shí),盡管連接已經(jīng)建立,還要為數(shù)據(jù)的收發(fā)做一些準(zhǔn)備。這些準(zhǔn)備中的第一步就是檢查 HTTP 響應(yīng):

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .void checkResponse(Response response) throws ProtocolException {if (response.code() != 101) {throw new ProtocolException("Expected HTTP 101 response but was '"+ response.code() + " " + response.message() + "'");}String headerConnection = response.header("Connection");if (!"Upgrade".equalsIgnoreCase(headerConnection)) {throw new ProtocolException("Expected 'Connection' header value 'Upgrade' but was '"+ headerConnection + "'");}String headerUpgrade = response.header("Upgrade");if (!"websocket".equalsIgnoreCase(headerUpgrade)) {throw new ProtocolException("Expected 'Upgrade' header value 'websocket' but was '" + headerUpgrade + "'");}String headerAccept = response.header("Sec-WebSocket-Accept");String acceptExpected = ByteString.encodeUtf8(key + WebSocketProtocol.ACCEPT_MAGIC).sha1().base64();if (!acceptExpected.equals(headerAccept)) {throw new ProtocolException("Expected 'Sec-WebSocket-Accept' header value '"+ acceptExpected + "' but was '" + headerAccept + "'");}}. . . . . .public void failWebSocket(Exception e, Response response) {Streams streamsToClose;synchronized (this) {if (failed) return; // Already failed.failed = true;streamsToClose = this.streams;this.streams = null;if (cancelFuture != null) cancelFuture.cancel(false);if (executor != null) executor.shutdown();}try {listener.onFailure(this, e, response);} finally {closeQuietly(streamsToClose);}}

    根據(jù) WebSocket 的協(xié)議,服務(wù)器端用如下響應(yīng),來(lái)表示接受建立 WebSocket 連接的請(qǐng)求:

  • 響應(yīng)碼是 101。
  • "Connection" header 的值為 "Upgrade",以表明服務(wù)器并沒(méi)有在處理完請(qǐng)求之后把連接個(gè)斷開(kāi)。
  • "Upgrade" header 的值為 "websocket",以表明服務(wù)器接受后面使用 WebSocket 來(lái)通信。
  • "Sec-WebSocket-Accept" header 的值為,key + WebSocketProtocol.ACCEPT_MAGIC 做 SHA1 hash,然后做 base64 編碼,來(lái)做服務(wù)器接受連接的驗(yàn)證。關(guān)于這部分的設(shè)計(jì)的詳細(xì)信息,可參考 WebSocket 協(xié)議規(guī)范。
  • 為數(shù)據(jù)收發(fā)做準(zhǔn)備的第二步是,初始化用于輸入輸出的 Source 和 Sink。Source 和 Sink 創(chuàng)建于之前發(fā)送HTTP請(qǐng)求的時(shí)候。這里會(huì)阻止在這個(gè)連接上再創(chuàng)建新的流。

    public final class RealConnection extends Http2Connection.Listener implements Connection {. . . . . .public RealWebSocket.Streams newWebSocketStreams(final StreamAllocation streamAllocation) {return new RealWebSocket.Streams(true, source, sink) {@Override public void close() throws IOException {streamAllocation.streamFinished(true, streamAllocation.codec());}};}

    Streams是一個(gè) BufferedSource 和 BufferedSink 的holder:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .public abstract static class Streams implements Closeable {public final boolean client;public final BufferedSource source;public final BufferedSink sink;public Streams(boolean client, BufferedSource source, BufferedSink sink) {this.client = client;this.source = source;this.sink = sink;}}

    第三步是調(diào)用回調(diào) onOpen()。

    第四步是初始化 Reader 和 Writer:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .public void initReaderAndWriter(String name, long pingIntervalMillis, Streams streams) throws IOException {synchronized (this) {this.streams = streams;this.writer = new WebSocketWriter(streams.client, streams.sink, random);this.executor = new ScheduledThreadPoolExecutor(1, Util.threadFactory(name, false));if (pingIntervalMillis != 0) {executor.scheduleAtFixedRate(new PingRunnable(), pingIntervalMillis, pingIntervalMillis, MILLISECONDS);}if (!messageAndCloseQueue.isEmpty()) {runWriter(); // Send messages that were enqueued before we were connected.}}reader = new WebSocketReader(streams.client, streams.source, this);}

    OkHttp使用 WebSocketReader 和 WebSocketWriter 來(lái)處理數(shù)據(jù)的收發(fā)。在發(fā)送數(shù)據(jù)時(shí)將數(shù)據(jù)組織成幀,在接收數(shù)據(jù)時(shí)則進(jìn)行反向擦做,同時(shí)處理 WebSocket 的控制消息。

    WebSocket 的所有數(shù)據(jù)發(fā)送動(dòng)作,都會(huì)在單線程線程池的線程中,通過(guò) WebSocketWriter 執(zhí)行。在這里會(huì)創(chuàng)建 ScheduledThreadPoolExecutor 用于跑數(shù)據(jù)的發(fā)送操作。WebSocket 協(xié)議中主要會(huì)傳輸兩種類型的幀,一是控制幀,主要是用于連接保活的 Ping 幀等;二是用戶數(shù)據(jù)載荷幀。在這里會(huì)根據(jù)用戶的配置,調(diào)度 Ping 幀周期性地發(fā)送。我們?cè)谡{(diào)用 WebSocket 的接口發(fā)送數(shù)據(jù)時(shí),數(shù)據(jù)并不是同步發(fā)送的,而是被放在了一個(gè)消息隊(duì)列中。發(fā)送消息的 Runnable 從消息隊(duì)列中讀取數(shù)據(jù)發(fā)送。這里會(huì)檢查消息隊(duì)列中是否有數(shù)據(jù),如果有的話,會(huì)調(diào)度發(fā)送消息的 Runnable 執(zhí)行。

    第五步是配置socket的超時(shí)時(shí)間為0,也就是阻塞IO。

    第六步執(zhí)行 loopReader()。這實(shí)際上是進(jìn)入了消息讀取循環(huán)了,也就是數(shù)據(jù)接收的邏輯了。

    數(shù)據(jù)發(fā)送

    我們可以通過(guò) WebSocket 接口的 send(String text) 和 send(ByteString bytes) 分別發(fā)送文本的和二進(jìn)制格式的消息。

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .@Override public boolean send(String text) {if (text == null) throw new NullPointerException("text == null");return send(ByteString.encodeUtf8(text), OPCODE_TEXT);}@Override public boolean send(ByteString bytes) {if (bytes == null) throw new NullPointerException("bytes == null");return send(bytes, OPCODE_BINARY);}private synchronized boolean send(ByteString data, int formatOpcode) {// Don't send new frames after we've failed or enqueued a close frame.if (failed || enqueuedClose) return false;// If this frame overflows the buffer, reject it and close the web socket.if (queueSize + data.size() > MAX_QUEUE_SIZE) {close(CLOSE_CLIENT_GOING_AWAY, null);return false;}// Enqueue the message frame.queueSize += data.size();messageAndCloseQueue.add(new Message(formatOpcode, data));runWriter();return true;}. . . . . .private void runWriter() {assert (Thread.holdsLock(this));if (executor != null) {executor.execute(writerRunnable);}}

    可以看到我們調(diào)用發(fā)送數(shù)據(jù)的接口時(shí),做的事情主要是將數(shù)據(jù)格式化,構(gòu)造消息,放進(jìn)一個(gè)消息隊(duì)列,然后調(diào)度 writerRunnable 執(zhí)行。

    此外,值得注意的是,當(dāng)消息隊(duì)列中的未發(fā)送數(shù)據(jù)超出最大大小限制,WebSocket 連接會(huì)被直接關(guān)閉。對(duì)于發(fā)送失敗過(guò)或被關(guān)閉了的 WebSocket,將無(wú)法再發(fā)送信息。

    在 writerRunnable 中會(huì)循環(huán)調(diào)用 writeOneFrame() 逐幀發(fā)送數(shù)據(jù),直到數(shù)據(jù)發(fā)完,或發(fā)送失敗。在 WebSocket 協(xié)議中,客戶端需要發(fā)送 四種類型 的幀:

  • PING 幀
  • PONG 幀
  • CLOSE 幀
  • MESSAGE 幀
  • PING幀用于連接保活,它的發(fā)送是在 PingRunnable 中執(zhí)行的,在初始化 Reader 和 Writer 的時(shí)候,就會(huì)根據(jù)設(shè)置調(diào)度執(zhí)行或不執(zhí)行。除PING 幀外的其它 三種 幀,都在 writeOneFrame() 中發(fā)送。PONG 幀是對(duì)服務(wù)器發(fā)過(guò)來(lái)的 PING 幀的響應(yīng),同樣用于保活連接。后面我們?cè)诜治鲞B接的保活時(shí)會(huì)更詳細(xì)的分析 PING 和 PONG 這兩種幀。CLOSE 幀用于關(guān)閉連接,稍后我們?cè)诜治鲞B接關(guān)閉過(guò)程時(shí)再來(lái)詳細(xì)地分析。

    這里我們主要關(guān)注用戶數(shù)據(jù)發(fā)送的部分。PONG 幀具有最高的發(fā)送優(yōu)先級(jí)。在沒(méi)有PONG 幀需要發(fā)送時(shí),writeOneFrame() 從消息隊(duì)列中取出一條消息,如果消息不是 CLOSE 幀,則主要通過(guò)如下的過(guò)程進(jìn)行發(fā)送:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .boolean writeOneFrame() throws IOException {WebSocketWriter writer;ByteString pong;Object messageOrClose = null;int receivedCloseCode = -1;String receivedCloseReason = null;Streams streamsToClose = null;synchronized (RealWebSocket.this) {if (failed) {return false; // Failed web socket.}writer = this.writer;pong = pongQueue.poll();if (pong == null) {messageOrClose = messageAndCloseQueue.poll();. . . . . .} else if (messageOrClose instanceof Message) {ByteString data = ((Message) messageOrClose).data;BufferedSink sink = Okio.buffer(writer.newMessageSink(((Message) messageOrClose).formatOpcode, data.size()));sink.write(data);sink.close();synchronized (this) {queueSize -= data.size();}} else if (messageOrClose instanceof Close) {

    數(shù)據(jù)發(fā)送的過(guò)程可以總結(jié)如下:

  • 創(chuàng)建一個(gè) BufferedSink 用于數(shù)據(jù)發(fā)送。
  • 將數(shù)據(jù)寫(xiě)入前面創(chuàng)建的 BufferedSink 中。
  • 關(guān)閉 BufferedSink。
  • 更新 queueSize 以正確地指示未發(fā)送數(shù)據(jù)的長(zhǎng)度。
  • 這里面的玄機(jī)主要在創(chuàng)建的 BufferedSink。創(chuàng)建的 Sink 是一個(gè) FrameSink:

    static void toggleMask(byte[] buffer, long byteCount, byte[] key, long frameBytesRead) {int keyLength = key.length;for (int i = 0; i < byteCount; i++, frameBytesRead++) {int keyIndex = (int) (frameBytesRead % keyLength);buffer[i] = (byte) (buffer[i] ^ key[keyIndex]);}}. . . . . .Sink newMessageSink(int formatOpcode, long contentLength) {if (activeWriter) {throw new IllegalStateException("Another message writer is active. Did you call close()?");}activeWriter = true;// Reset FrameSink state for a new writer.frameSink.formatOpcode = formatOpcode;frameSink.contentLength = contentLength;frameSink.isFirstFrame = true;frameSink.closed = false;return frameSink;}void writeMessageFrameSynchronized(int formatOpcode, long byteCount, boolean isFirstFrame,boolean isFinal) throws IOException {assert Thread.holdsLock(this);if (writerClosed) throw new IOException("closed");int b0 = isFirstFrame ? formatOpcode : OPCODE_CONTINUATION;if (isFinal) {b0 |= B0_FLAG_FIN;}sink.writeByte(b0);int b1 = 0;if (isClient) {b1 |= B1_FLAG_MASK;}if (byteCount <= PAYLOAD_BYTE_MAX) {b1 |= (int) byteCount;sink.writeByte(b1);} else if (byteCount <= PAYLOAD_SHORT_MAX) {b1 |= PAYLOAD_SHORT;sink.writeByte(b1);sink.writeShort((int) byteCount);} else {b1 |= PAYLOAD_LONG;sink.writeByte(b1);sink.writeLong(byteCount);}if (isClient) {random.nextBytes(maskKey);sink.write(maskKey);for (long written = 0; written < byteCount; ) {int toRead = (int) Math.min(byteCount, maskBuffer.length);int read = buffer.read(maskBuffer, 0, toRead);if (read == -1) throw new AssertionError();toggleMask(maskBuffer, read, maskKey, written);sink.write(maskBuffer, 0, read);written += read;}} else {sink.write(buffer, byteCount);}sink.emit();}final class FrameSink implements Sink {int formatOpcode;long contentLength;boolean isFirstFrame;boolean closed;@Override public void write(Buffer source, long byteCount) throws IOException {if (closed) throw new IOException("closed");buffer.write(source, byteCount);// Determine if this is a buffered write which we can defer until close() flushes.boolean deferWrite = isFirstFrame&& contentLength != -1&& buffer.size() > contentLength - 8192 /* segment size */;long emitCount = buffer.completeSegmentByteCount();if (emitCount > 0 && !deferWrite) {synchronized (WebSocketWriter.this) {writeMessageFrameSynchronized(formatOpcode, emitCount, isFirstFrame, false /* final */);}isFirstFrame = false;}}@Override public void flush() throws IOException {if (closed) throw new IOException("closed");synchronized (WebSocketWriter.this) {writeMessageFrameSynchronized(formatOpcode, buffer.size(), isFirstFrame, false /* final */);}isFirstFrame = false;}@Override public Timeout timeout() {return sink.timeout();}@SuppressWarnings("PointlessBitwiseExpression")@Override public void close() throws IOException {if (closed) throw new IOException("closed");synchronized (WebSocketWriter.this) {writeMessageFrameSynchronized(formatOpcode, buffer.size(), isFirstFrame, true /* final */);}closed = true;activeWriter = false;}}

    FrameSink 的 write() 會(huì)先將數(shù)據(jù)寫(xiě)如一個(gè) Buffer 中,然后再?gòu)倪@個(gè) Buffer 中讀取數(shù)據(jù)來(lái)發(fā)送。如果是第一次發(fā)送數(shù)據(jù),同時(shí)剩余要發(fā)送的數(shù)據(jù)小于 8192 字節(jié)時(shí),會(huì)延遲執(zhí)行實(shí)際的數(shù)據(jù)發(fā)送,等 close() 時(shí)刷新。根據(jù) RealWebSocket 的 writeOneFrame() 的邏輯,在 write() 時(shí),總是寫(xiě)入整個(gè)消息的所有數(shù)據(jù),因而,在 FrameSink 的 write() 中總是不會(huì)發(fā)送數(shù)據(jù)的。

    writeMessageFrameSynchronized() 將用戶數(shù)據(jù)格式化并發(fā)送出去。規(guī)范中定義的數(shù)據(jù)格式如下:

    0 1 2 30 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1+-+-+-+-+-------+-+-------------+-------------------------------+|F|R|R|R| opcode|M| Payload len | Extended payload length ||I|S|S|S| (4) |A| (7) | (16/64) ||N|V|V|V| |S| | (if payload len==126/127) || |1|2|3| |K| | |+-+-+-+-+-------+-+-------------+ - - - - - - - - - - - - - - - +| Extended payload length continued, if payload len == 127 |+ - - - - - - - - - - - - - - - +-------------------------------+| |Masking-key, if MASK set to 1 |+-------------------------------+-------------------------------+| Masking-key (continued) | Payload Data |+-------------------------------- - - - - - - - - - - - - - - - +: Payload Data continued ... :+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +| Payload Data continued ... |+---------------------------------------------------------------+

    基本結(jié)構(gòu)為:

  • 第一個(gè)字節(jié)是 meta data 控制位,包括四位的操作碼,用于指明這是否是消息的最后一幀的FIN位及三個(gè)保留位。
  • 第二個(gè)字節(jié)包括掩碼位,和載荷長(zhǎng)度或載荷長(zhǎng)度指示。只有載荷長(zhǎng)度比較小,在 127 以內(nèi)時(shí),載荷長(zhǎng)度才會(huì)包含在這個(gè)字節(jié)中。否則這個(gè)字節(jié)中將包含載荷長(zhǎng)度指示的位。
  • 可選的載荷長(zhǎng)度。載荷長(zhǎng)度大于127時(shí),幀中會(huì)專門有一些字節(jié)來(lái)描述載荷的長(zhǎng)度。載荷長(zhǎng)度具體占用幾個(gè)自己,因載荷的實(shí)際長(zhǎng)度而異。
  • 可選的掩碼字節(jié)。客戶端發(fā)送的幀,設(shè)置掩碼指示位,并包含四個(gè)字節(jié)的掩碼字節(jié)。
  • 載荷數(shù)據(jù)。客戶端發(fā)送的數(shù)據(jù),會(huì)將原始的數(shù)據(jù)與掩碼字節(jié)做異或之后再發(fā)送。
  • 關(guān)于幀格式的更詳細(xì)信息,可以參考 WebSocket Protocol 規(guī)范。

    數(shù)據(jù)的接收

    如我們前面看到的, 在握手的HTTP請(qǐng)求返回之后,會(huì)在HTTP請(qǐng)求的回調(diào)里,啟動(dòng)消息讀取循環(huán) loopReader():

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . ./** Receive frames until there are no more. Invoked only by the reader thread. */public void loopReader() throws IOException {while (receivedCloseCode == -1) {// This method call results in one or more onRead* methods being called on this thread.reader.processNextFrame();}}

    在這個(gè)循環(huán)中,不斷通過(guò) WebSocketReader 的 processNextFrame() 讀取消息,直到收到了關(guān)閉連接的消息。

    final class WebSocketReader {public interface FrameCallback {void onReadMessage(String text) throws IOException;void onReadMessage(ByteString bytes) throws IOException;void onReadPing(ByteString buffer);void onReadPong(ByteString buffer);void onReadClose(int code, String reason);}. . . . . .void processNextFrame() throws IOException {readHeader();if (isControlFrame) {readControlFrame();} else {readMessageFrame();}}private void readHeader() throws IOException {if (closed) throw new IOException("closed");// Disable the timeout to read the first byte of a new frame.int b0;long timeoutBefore = source.timeout().timeoutNanos();source.timeout().clearTimeout();try {b0 = source.readByte() & 0xff;} finally {source.timeout().timeout(timeoutBefore, TimeUnit.NANOSECONDS);}opcode = b0 & B0_MASK_OPCODE;isFinalFrame = (b0 & B0_FLAG_FIN) != 0;isControlFrame = (b0 & OPCODE_FLAG_CONTROL) != 0;// Control frames must be final frames (cannot contain continuations).if (isControlFrame && !isFinalFrame) {throw new ProtocolException("Control frames must be final.");}boolean reservedFlag1 = (b0 & B0_FLAG_RSV1) != 0;boolean reservedFlag2 = (b0 & B0_FLAG_RSV2) != 0;boolean reservedFlag3 = (b0 & B0_FLAG_RSV3) != 0;if (reservedFlag1 || reservedFlag2 || reservedFlag3) {// Reserved flags are for extensions which we currently do not support.throw new ProtocolException("Reserved flags are unsupported.");}int b1 = source.readByte() & 0xff;isMasked = (b1 & B1_FLAG_MASK) != 0;if (isMasked == isClient) {// Masked payloads must be read on the server. Unmasked payloads must be read on the client.throw new ProtocolException(isClient? "Server-sent frames must not be masked.": "Client-sent frames must be masked.");}// Get frame length, optionally reading from follow-up bytes if indicated by special values.frameLength = b1 & B1_MASK_LENGTH;if (frameLength == PAYLOAD_SHORT) {frameLength = source.readShort() & 0xffffL; // Value is unsigned.} else if (frameLength == PAYLOAD_LONG) {frameLength = source.readLong();if (frameLength < 0) {throw new ProtocolException("Frame length 0x" + Long.toHexString(frameLength) + " > 0x7FFFFFFFFFFFFFFF");}}frameBytesRead = 0;if (isControlFrame && frameLength > PAYLOAD_BYTE_MAX) {throw new ProtocolException("Control frame must be less than " + PAYLOAD_BYTE_MAX + "B.");}if (isMasked) {// Read the masking key as bytes so that they can be used directly for unmasking.source.readFully(maskKey);}}

    processNextFrame() 先讀取 Header 的兩個(gè)字節(jié),然后根據(jù) Header 的信息,讀取數(shù)據(jù)內(nèi)容。

    在讀取 Header 時(shí),讀的第一個(gè)字節(jié)是同步的不計(jì)超時(shí)時(shí)間的。WebSocketReader 從 Header 中,獲取到這個(gè)幀是不是消息的最后一幀,消息的類型,是否有掩碼字節(jié),保留位,幀的長(zhǎng)度,以及掩碼字節(jié)等信息。WebSocket 通過(guò)掩碼位和掩碼字節(jié)來(lái)區(qū)分?jǐn)?shù)據(jù)是從客戶端發(fā)送給服務(wù)器的,還是服務(wù)器發(fā)送給客戶端的。這里會(huì)根據(jù)協(xié)議,對(duì)這些信息進(jìn)行有效性一致性檢驗(yàn),若不一致則會(huì)拋出 ProtocolException。

    WebSocketReader 同步讀取時(shí)的調(diào)用棧如下:


    Reader Thread

    通過(guò)幀的 Header 確定了是數(shù)據(jù)幀,則會(huì)執(zhí)行 readMessageFrame() 讀取消息幀:

    final class WebSocketReader {. . . . . .private void readMessageFrame() throws IOException {int opcode = this.opcode;if (opcode != OPCODE_TEXT && opcode != OPCODE_BINARY) {throw new ProtocolException("Unknown opcode: " + toHexString(opcode));}Buffer message = new Buffer();readMessage(message);if (opcode == OPCODE_TEXT) {frameCallback.onReadMessage(message.readUtf8());} else {frameCallback.onReadMessage(message.readByteString());}}/** Read headers and process any control frames until we reach a non-control frame. */void readUntilNonControlFrame() throws IOException {while (!closed) {readHeader();if (!isControlFrame) {break;}readControlFrame();}}/*** Reads a message body into across one or more frames. Control frames that occur between* fragments will be processed. If the message payload is masked this will unmask as it's being* processed.*/private void readMessage(Buffer sink) throws IOException {while (true) {if (closed) throw new IOException("closed");if (frameBytesRead == frameLength) {if (isFinalFrame) return; // We are exhausted and have no continuations.readUntilNonControlFrame();if (opcode != OPCODE_CONTINUATION) {throw new ProtocolException("Expected continuation opcode. Got: " + toHexString(opcode));}if (isFinalFrame && frameLength == 0) {return; // Fast-path for empty final frame.}}long toRead = frameLength - frameBytesRead;long read;if (isMasked) {toRead = Math.min(toRead, maskBuffer.length);read = source.read(maskBuffer, 0, (int) toRead);if (read == -1) throw new EOFException();toggleMask(maskBuffer, read, maskKey, frameBytesRead);sink.write(maskBuffer, 0, (int) read);} else {read = source.read(sink, toRead);if (read == -1) throw new EOFException();}frameBytesRead += read;}}

    這個(gè)過(guò)程中,會(huì)讀取一條消息包含的所有數(shù)據(jù)幀。按照 WebSocket 的標(biāo)準(zhǔn),包含用戶數(shù)據(jù)的消息數(shù)據(jù)幀可以和控制幀交替發(fā)送;但消息之間的數(shù)據(jù)幀不可以。因而在這個(gè)過(guò)程中,若遇到了控制幀,則會(huì)先讀取控制幀進(jìn)行處理,然后繼續(xù)讀取消息的數(shù)據(jù)幀,直到讀取了消息的所有數(shù)據(jù)幀。

    掩碼位和掩碼字節(jié),對(duì)于客戶端而言,發(fā)送的數(shù)據(jù)中包含這些東西,在接收的數(shù)據(jù)中不包含這些;對(duì)于服務(wù)器而言,則是在接收的數(shù)據(jù)中包含這些,發(fā)送的數(shù)據(jù)中不包含。OkHttp 既支持服務(wù)器開(kāi)發(fā),也支持客戶端開(kāi)發(fā),因而可以看到對(duì)于掩碼位和掩碼字節(jié)完整的處理。

    在一個(gè)消息讀取完成之后,會(huì)通過(guò)回調(diào) FrameCallback 將讀取的內(nèi)容通知出去。

    final class WebSocketReader {. . . . . .WebSocketReader(boolean isClient, BufferedSource source, FrameCallback frameCallback) {if (source == null) throw new NullPointerException("source == null");if (frameCallback == null) throw new NullPointerException("frameCallback == null");this.isClient = isClient;this.source = source;this.frameCallback = frameCallback;}

    這一事件會(huì)通知到 RealWebSocket。

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .@Override public void onReadMessage(String text) throws IOException {listener.onMessage(this, text);}@Override public void onReadMessage(ByteString bytes) throws IOException {listener.onMessage(this, bytes);}

    在 RealWebSocket 中,這一事件又被通知到我們?cè)趹?yīng)用程序中創(chuàng)建的回調(diào) WebSocketListener。

    連接的保活

    連接的保活通過(guò) PING 幀和 PONG 幀來(lái)實(shí)現(xiàn)。如我們前面看到的,若用戶設(shè)置了 PING 幀的發(fā)送周期,在握手的HTTP請(qǐng)求返回時(shí),消息讀取循環(huán)開(kāi)始前會(huì)調(diào)度 PingRunnable 周期性的向服務(wù)器發(fā)送 PING 幀:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .private final class PingRunnable implements Runnable {PingRunnable() {}@Override public void run() {writePingFrame();}}void writePingFrame() {WebSocketWriter writer;synchronized (this) {if (failed) return;writer = this.writer;}try {writer.writePing(ByteString.EMPTY);} catch (IOException e) {failWebSocket(e, null);}}

    在 PingRunnable 中,通過(guò) WebSocketWriter 發(fā)送 PING 幀:

    final class WebSocketWriter {. . . . . ./** Send a ping with the supplied {@code payload}. */void writePing(ByteString payload) throws IOException {synchronized (this) {writeControlFrameSynchronized(OPCODE_CONTROL_PING, payload);}}. . . . . .private void writeControlFrameSynchronized(int opcode, ByteString payload) throws IOException {assert Thread.holdsLock(this);if (writerClosed) throw new IOException("closed");int length = payload.size();if (length > PAYLOAD_BYTE_MAX) {throw new IllegalArgumentException("Payload size must be less than or equal to " + PAYLOAD_BYTE_MAX);}int b0 = B0_FLAG_FIN | opcode;sink.writeByte(b0);int b1 = length;if (isClient) {b1 |= B1_FLAG_MASK;sink.writeByte(b1);random.nextBytes(maskKey);sink.write(maskKey);byte[] bytes = payload.toByteArray();toggleMask(bytes, bytes.length, maskKey, 0);sink.write(bytes);} else {sink.writeByte(b1);sink.write(payload);}sink.flush();}

    PING 幀是一個(gè)不包含載荷的控制幀。關(guān)于掩碼位和掩碼字節(jié)的設(shè)置,與消息的數(shù)據(jù)幀相同。即客戶端發(fā)送的幀,設(shè)置掩碼位,幀中包含掩碼字節(jié);服務(wù)器發(fā)送的幀,不設(shè)置掩碼位,幀中不包含掩碼字節(jié)。

    通過(guò) WebSocket 通信的雙方,在收到對(duì)方發(fā)來(lái)的 PING 幀時(shí),需要用PONG幀來(lái)回復(fù)。在 WebSocketReader 的 readControlFrame() 中可以看到這一點(diǎn):

    final class WebSocketReader {. . . . . .private void readControlFrame() throws IOException {Buffer buffer = new Buffer();if (frameBytesRead < frameLength) {if (isClient) {source.readFully(buffer, frameLength);} else {while (frameBytesRead < frameLength) {int toRead = (int) Math.min(frameLength - frameBytesRead, maskBuffer.length);int read = source.read(maskBuffer, 0, toRead);if (read == -1) throw new EOFException();toggleMask(maskBuffer, read, maskKey, frameBytesRead);buffer.write(maskBuffer, 0, read);frameBytesRead += read;}}}switch (opcode) {case OPCODE_CONTROL_PING:frameCallback.onReadPing(buffer.readByteString());break;case OPCODE_CONTROL_PONG:frameCallback.onReadPong(buffer.readByteString());break;

    PING 幀和 PONG 幀都不帶載荷,控制幀讀寫(xiě)時(shí)對(duì)于載荷長(zhǎng)度的處理,都是為 CLOSE 幀做的。因而針對(duì) PING 幀和 PONG 幀,除了 Header 外, readControlFrame() 實(shí)際上無(wú)需再讀取任何數(shù)據(jù),但它會(huì)將這些事件通知出去:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .@Override public synchronized void onReadPing(ByteString payload) {// Don't respond to pings after we've failed or sent the close frame.if (failed || (enqueuedClose && messageAndCloseQueue.isEmpty())) return;pongQueue.add(payload);runWriter();pingCount++;}@Override public synchronized void onReadPong(ByteString buffer) {// This API doesn't expose pings.pongCount++;}

    可見(jiàn)在收到 PING 幀的時(shí)候,總是會(huì)發(fā)一個(gè) PONG 幀出去,且通常其沒(méi)有載荷數(shù)據(jù)。在收到一個(gè) PONG 幀時(shí),則通常只是記錄一下,然后什么也不做。如我們前面所見(jiàn),PONG 幀在 writerRunnable 中被發(fā)送出去:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .if (pong != null) {writer.writePong(pong);} else if (messageOrClose instanceof Message) {

    PONG 幀的發(fā)送與 PING 幀的非常相似:

    final class WebSocketWriter {. . . . . ./** Send a pong with the supplied {@code payload}. */void writePong(ByteString payload) throws IOException {synchronized (this) {writeControlFrameSynchronized(OPCODE_CONTROL_PONG, payload);}}

    連接的關(guān)閉

    連接的關(guān)閉,與數(shù)據(jù)發(fā)送的過(guò)程頗有幾分相似之處。通過(guò) WebSocket 接口的 close(int code, String reason) 我們可以關(guān)閉一個(gè) WebSocket 連接:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .@Override public boolean close(int code, String reason) {return close(code, reason, CANCEL_AFTER_CLOSE_MILLIS);}synchronized boolean close(int code, String reason, long cancelAfterCloseMillis) {validateCloseCode(code);ByteString reasonBytes = null;if (reason != null) {reasonBytes = ByteString.encodeUtf8(reason);if (reasonBytes.size() > CLOSE_MESSAGE_MAX) {throw new IllegalArgumentException("reason.size() > " + CLOSE_MESSAGE_MAX + ": " + reason);}}if (failed || enqueuedClose) return false;// Immediately prevent further frames from being enqueued.enqueuedClose = true;// Enqueue the close frame.messageAndCloseQueue.add(new Close(code, reasonBytes, cancelAfterCloseMillis));runWriter();return true;}

    在執(zhí)行關(guān)閉連接動(dòng)作前,會(huì)先檢查一下 close code 的有效性在合法范圍內(nèi)。關(guān)于不同 close code 的詳細(xì)說(shuō)明,可以參考 WebSocket 協(xié)議規(guī)范。

    檢查完了之后,會(huì)構(gòu)造一個(gè) Close 消息放入發(fā)送消息隊(duì)列,并調(diào)度 writerRunnable 執(zhí)行。Close 消息可以帶有不超出 123 字節(jié)的字符串,以作為 Close message,來(lái)說(shuō)明連接關(guān)閉的原因。

    連接的關(guān)閉分為主動(dòng)關(guān)閉和被動(dòng)關(guān)閉。客戶端先向服務(wù)器發(fā)送一個(gè) CLOSE 幀,然后服務(wù)器恢復(fù)一個(gè) CLOSE 幀,對(duì)于客戶端而言,這個(gè)過(guò)程為主動(dòng)關(guān)閉;反之則為對(duì)客戶端而言則為被動(dòng)關(guān)閉。

    在 writerRunnable 執(zhí)行的 writeOneFrame() 實(shí)際發(fā)送 CLOSE 幀:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .messageOrClose = messageAndCloseQueue.poll();if (messageOrClose instanceof Close) {receivedCloseCode = this.receivedCloseCode;receivedCloseReason = this.receivedCloseReason;if (receivedCloseCode != -1) {streamsToClose = this.streams;this.streams = null;this.executor.shutdown();} else {// When we request a graceful close also schedule a cancel of the websocket.cancelFuture = executor.schedule(new CancelRunnable(),((Close) messageOrClose).cancelAfterCloseMillis, MILLISECONDS);}} else if (messageOrClose == null) {return false; // The queue is exhausted.}}. . . . . .} else if (messageOrClose instanceof Close) {Close close = (Close) messageOrClose;writer.writeClose(close.code, close.reason);// We closed the writer: now both reader and writer are closed.if (streamsToClose != null) {listener.onClosed(this, receivedCloseCode, receivedCloseReason);}} else {

    發(fā)送 CLOSE 幀也分為主動(dòng)關(guān)閉的發(fā)送還是被動(dòng)關(guān)閉的發(fā)送。
    對(duì)于被動(dòng)關(guān)閉,在發(fā)送完 CLOSE 幀之后,連接被最終關(guān)閉,因而,發(fā)送 CLOSE 幀之前,這里會(huì)停掉發(fā)送消息用的 executor。而在發(fā)送之后,則會(huì)通過(guò) onClosed() 通知用戶。

    而對(duì)于主動(dòng)關(guān)閉,則在發(fā)送前會(huì)調(diào)度 CancelRunnable 的執(zhí)行,發(fā)送后不會(huì)通過(guò) onClosed() 通知用戶。

    final class WebSocketWriter {. . . . . .void writeClose(int code, ByteString reason) throws IOException {ByteString payload = ByteString.EMPTY;if (code != 0 || reason != null) {if (code != 0) {validateCloseCode(code);}Buffer buffer = new Buffer();buffer.writeShort(code);if (reason != null) {buffer.write(reason);}payload = buffer.readByteString();}synchronized (this) {try {writeControlFrameSynchronized(OPCODE_CONTROL_CLOSE, payload);} finally {writerClosed = true;}}}

    將 CLOSE 幀發(fā)送到網(wǎng)絡(luò)的過(guò)程與 PING 和 PONG 幀的頗為相似,僅有的差別就是 CLOSE 幀有載荷。關(guān)于掩碼位和掩碼自己的規(guī)則,同樣適用于 CLOSE 幀的發(fā)送。

    CLOSE 的讀取在 WebSocketReader 的 readControlFrame()中:

    final class WebSocketReader {. . . . . .private void readControlFrame() throws IOException {Buffer buffer = new Buffer();if (frameBytesRead < frameLength) {if (isClient) {source.readFully(buffer, frameLength);} else {while (frameBytesRead < frameLength) {int toRead = (int) Math.min(frameLength - frameBytesRead, maskBuffer.length);int read = source.read(maskBuffer, 0, toRead);if (read == -1) throw new EOFException();toggleMask(maskBuffer, read, maskKey, frameBytesRead);buffer.write(maskBuffer, 0, read);frameBytesRead += read;}}}switch (opcode) {. . . . . .case OPCODE_CONTROL_CLOSE:int code = CLOSE_NO_STATUS_CODE;String reason = "";long bufferSize = buffer.size();if (bufferSize == 1) {throw new ProtocolException("Malformed close payload length of 1.");} else if (bufferSize != 0) {code = buffer.readShort();reason = buffer.readUtf8();String codeExceptionMessage = WebSocketProtocol.closeCodeExceptionMessage(code);if (codeExceptionMessage != null) throw new ProtocolException(codeExceptionMessage);}frameCallback.onReadClose(code, reason);closed = true;break;default:throw new ProtocolException("Unknown control opcode: " + toHexString(opcode));}}

    讀到 CLOSE 幀時(shí),WebSocketReader 會(huì)將這一事件通知出去:

    public final class RealWebSocket implements WebSocket, WebSocketReader.FrameCallback {. . . . . .@Override public void onReadClose(int code, String reason) {if (code == -1) throw new IllegalArgumentException();Streams toClose = null;synchronized (this) {if (receivedCloseCode != -1) throw new IllegalStateException("already closed");receivedCloseCode = code;receivedCloseReason = reason;if (enqueuedClose && messageAndCloseQueue.isEmpty()) {toClose = this.streams;this.streams = null;if (cancelFuture != null) cancelFuture.cancel(false);this.executor.shutdown();}}try {listener.onClosing(this, code, reason);if (toClose != null) {listener.onClosed(this, code, reason);}} finally {closeQuietly(toClose);}}

    對(duì)于收到的 CLOSE 幀處理同樣分為主動(dòng)關(guān)閉的情況和被動(dòng)關(guān)閉的情況。與 CLOSE 發(fā)送時(shí)的情形正好相反,若是主動(dòng)關(guān)閉,則在收到 CLOSE 幀之后,WebSocket 連接最終斷開(kāi),因而需要停掉executor,被動(dòng)關(guān)閉則暫時(shí)不需要。

    收到 CLOSE 幀,總是會(huì)通過(guò) onClosing() 將事件通知出去。

    對(duì)于主動(dòng)關(guān)閉的情形,最后還會(huì)通過(guò) onClosed() 通知用戶,連接已經(jīng)最終關(guān)閉。

    關(guān)于 WebSocket 的 CLOSE 幀的更多說(shuō)明,可以參考 WebSocket協(xié)議規(guī)范。

    WebSocket連接的生命周期

    總結(jié)一下 WebSocket 連接的生命周期:

  • 連接通過(guò)一個(gè)HTTP請(qǐng)求握手并建立連接。WebSocket 連接可以理解為是通過(guò)HTTP請(qǐng)求建立的普通TCP連接。
  • WebSocket 做了二進(jìn)制分幀。WebSocket 連接中收發(fā)的數(shù)據(jù)以幀為單位。主要有用于連接保活的控制幀 PING 和 PONG,用于用戶數(shù)據(jù)發(fā)送的 MESSAGE 幀,和用于關(guān)閉連接的控制幀 CLOSE。
  • 連接建立之后,通過(guò) PING 幀和 PONG 幀做連接保活。
  • 一次 send 數(shù)據(jù),被封為一個(gè)消息,通過(guò)一個(gè)或多個(gè) MESSAGE幀進(jìn)行發(fā)送。一個(gè)消息的幀和控制幀可以交叉發(fā)送,不同消息的幀之間不可以。
  • WebSocket 連接的兩端相互發(fā)送一個(gè) CLOSE 幀以最終關(guān)閉連接。
  • 關(guān)于 WebSocket 的詳細(xì)信息,可以參考 WebSocket協(xié)議規(guī)范。

    參考文檔

    WebSocket 協(xié)議規(guī)范
    WebSocket 實(shí)戰(zhàn)
    使用 HTML5 WebSocket 構(gòu)建實(shí)時(shí) Web 應(yīng)用
    WebSocket Client Example with OkHttp

    總結(jié)

    以上是生活随笔為你收集整理的OkHttp实现分析之Websocket的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    91精选| 久久免费视频一区 | 欧美日韩一区二区免费在线观看 | 国产精品久久久久久久久免费看 | 国产一区二区三区四区大秀 | 99精品在线免费 | 亚洲午夜精品久久久 | 九九九九九九精品任你躁 | 精品在线观 | 91av欧美| 欧美性色综合 | 中文一区二区三区在线观看 | 日韩在线电影观看 | 久久怡红院| 天天爽夜夜爽人人爽曰av | 亚洲涩综合 | 午夜视频亚洲 | 久福利 | 精品久久久国产 | 黄色电影网站在线观看 | 国产精品免费久久久久 | 中字幕视频在线永久在线观看免费 | 日韩a欧美 | 久久伊人精品一区二区三区 | av免费福利| 久久精品日本啪啪涩涩 | 在线小视频 | 色在线免费 | 黄色av网站在线免费观看 | 91麻豆精品国产91久久久无限制版 | 99久久精品午夜一区二区小说 | 欧美大香线蕉线伊人久久 | 在线观看亚洲 | 中文字幕一区av | 手机看片中文字幕 | 视频在线观看日韩 | 久久久久免费电影 | 国产成人亚洲精品自产在线 | 欧美狠狠操 | 99精品久久99久久久久 | 精品在线一区二区 | 久久国产精品99久久久久久老狼 | 欧美精品一二 | 国产在线久草 | 久久视频在线免费观看 | av超碰免费在线 | 亚洲欧洲中文日韩久久av乱码 | 日本精品中文字幕在线观看 | 欧美99热 | 四虎影视成人永久免费观看亚洲欧美 | 亚洲精品动漫成人3d无尽在线 | 日韩色在线观看 | 天天搞夜夜骑 | 日韩在线视频二区 | 久久在线免费视频 | 91视频大全| 亚洲精选在线 | 人人爱人人爽 | 亚洲精品xxx | 欧美精品久久 | 久草综合在线 | 四虎在线永久免费观看 | 国产精品久久久久久久久久久久午夜 | 亚洲精品小视频在线观看 | www.成人sex | 久久综合狠狠综合久久综合88 | 99re6热在线精品视频 | 国产五月色婷婷六月丁香视频 | 最近日本中文字幕 | 亚洲欧美国产精品 | 天天草天天 | 午夜精品一区二区三区在线 | www.在线观看视频 | 日本精品免费看 | 色综合天天综合网国产成人网 | 精品一二 | 亚洲特级毛片 | 成 人 a v天堂 | 999久久久久久 | 欧美做受69 | 欧美成人精品xxx | 亚洲成人黄色在线 | 天天插天天狠天天透 | 99精品观看 | 亚洲在线不卡 | 麻豆一区二区 | 久久免费视频在线观看30 | 国产一区二区播放 | 日韩中文字幕免费看 | 在线av资源| 婷婷久久网 | av线上免费看 | 91精品1区 | 激情在线网站 | 久久精品五月 | 久久a级片 | 狠狠干 狠狠操 | 国色天香永久免费 | 国产精品9区 | 黄色小网站在线观看 | 在线观看你懂的网址 | 蜜桃av观看| 99免费在线观看 | 中文字幕在线资源 | 久久99精品国产麻豆宅宅 | 午夜视频不卡 | 波多野结衣资源 | 精品在线一区二区 | 91精品视频免费看 | 国产区精品在线 | 国产美腿白丝袜足在线av | 国产精品美女久久久久久久久久久 | 亚州精品视频 | 99久久精品免费一区 | 黄色网www| 欧美午夜精品久久久久久浪潮 | 成人亚洲精品国产www | 在线观看网站你懂的 | 激情丁香5月 | 久久久在线免费观看 | 亚洲美女在线一区 | 久久涩视频 | 国产精品自在线拍国产 | 欧美精品少妇xxxxx喷水 | 久久久久久伊人 | 久久久wwww| 天天鲁天天干天天射 | 91av资源在线| 亚洲精品白浆高清久久久久久 | 成人av免费网站 | 日本亚洲国产 | 久久99热精品 | 亚洲欧洲精品一区二区 | 久久精品在线视频 | 亚洲精品三级 | 亚洲视频免费在线看 | 久久激情视频 久久 | 欧美一区二区日韩一区二区 | 亚洲h在线播放在线观看h | 精品国产aⅴ麻豆 | 日韩性xxxx| 美女av免费看 | 日日添夜夜添 | 国产欧美精品一区二区三区四区 | 一区在线观看视频 | 久久五月婷婷综合 | 国产剧情一区二区在线观看 | 久久婷婷一区 | 97在线超碰 | 8090yy亚洲精品久久 | 天天艹天天干天天 | 草久久久久| 91cn国产在线 | 在线观看亚洲免费视频 | 91精品国产网站 | 在线观看av片 | 日韩成人邪恶影片 | 国产123区在线观看 国产精品麻豆91 | 亚洲精品综合一区二区 | 亚洲影视九九影院在线观看 | 九九视频热 | 最近中文字幕完整高清 | 超碰在线最新网址 | 在线观看日韩精品视频 | 日本在线观看中文字幕 | 在线观看91精品国产网站 | 人人插人人艹 | 久久在线观看 | 天天爽天天爽天天爽 | 91九色国产蝌蚪 | 激情综合网五月 | 日日干夜夜草 | 中文字幕在线高清 | 国产 日韩 在线 亚洲 字幕 中文 | 99久久精品国产亚洲 | 国产成人精品a | 成人一级黄色片 | 精品久久久久久电影 | 久久久久久久久久久久久久免费看 | 九九三级毛片 | 亚洲天堂社区 | 五月婷婷综合在线观看 | 99精品在线直播 | 特级毛片网 | 手机成人av在线 | 国产成人一二三 | 97超碰人人澡 | 国产主播大尺度精品福利免费 | 日本精品一 | 伊人首页 | 精品视频亚洲 | 香蕉网在线观看 | 久久精品视频免费播放 | 亚洲精品美女久久 | 久久网址 | 91精品久久久久久综合五月天 | 四虎精品成人免费网站 | 免费看av在线 | 国产最新在线观看 | 亚洲天堂精品 | wwwwwww黄| 中文欧美字幕免费 | 国产黄色片免费在线观看 | 四虎成人精品永久免费av九九 | 国产91在线观| 波多野结衣最新 | 精品视频999 | 色网站在线| www国产在线 | 97在线视频免费播放 | 久久国产免费看 | 激情小说 五月 | 久久久久久久久久亚洲精品 | 色综合五月 | 在线黄色av电影 | 永久免费的啪啪网站免费观看浪潮 | 国产不卡高清 | 一区二区三区在线电影 | 日日夜夜天天 | 区一区二区三在线观看 | 91禁在线看 | 亚洲日韩中文字幕 | 97碰碰精品嫩模在线播放 | 黄色毛片一级片 | 国产999精品视频 | 99久久久久国产精品免费 | 国产高清免费视频 | 久久综合狠狠狠色97 | 综合精品久久 | 日本精品一区二区三区在线播放视频 | 视频在线一区 | 精品伊人久久久 | 久久久久欧美精品999 | 免费高清在线视频一区· | 蜜臀av网址| 午夜12点 | 午夜av在线播放 | 麻豆一区在线观看 | 成年人黄色大全 | 亚洲va欧美va国产va黑人 | 96久久| 久热这里有精品 | 手机av网站 | 日韩最新av | 麻花豆传媒一二三产区 | 亚洲精品视频免费在线观看 | 在线观看久久久久久 | 久久99精品热在线观看 | 色大片免费看 | 综合色播| 国产96在线观看 | 午夜精品久久久久久中宇69 | 韩国一区二区三区视频 | 亚洲精品久久久久久久蜜桃 | 天天色天天色天天色 | 麻豆国产视频下载 | 欧美电影黄色 | 97操操操| 免费av网站在线看 | 99久久精品国产一区二区成人 | 天天艹 | 久草在线费播放视频 | 天天草av| 久久av免费电影 | 在线观看岛国 | 国产美女搞久久 | 亚洲欧美国产精品 | 亚洲国产影院 | 国产精品麻豆免费版 | 深爱激情亚洲 | 成人av一级片 | 日韩高清一 | 久久精品国产免费看久久精品 | 丁香激情视频 | 亚洲成人精品在线观看 | 黄色精品国产 | 成人午夜电影在线 | 伊人影院99 | 亚洲在线免费视频 | 日本h视频在线观看 | 黄色资源网站 | 伊人色综合久久天天网 | 99久热在线精品视频 | 91在线国产观看 | 九九久久国产 | 96超碰在线 | 午夜色大片在线观看 | 97电影院在线观看 | 国产黄色av网站 | 久久人人爽 | 奇米影视在线99精品 | 国内精品久久久 | 91在线免费播放 | 亚洲国产精品久久久久 | 国产91学生粉嫩喷水 | 精品uu| 日韩xxxbbb| 在线视频婷婷 | 国产一区在线免费观看视频 | 久久在线免费观看 | 精品国产乱码久久久久久1区二区 | 欧美日韩视频精品 | 国产亚洲精品日韩在线tv黄 | 99色人| 西西444www | 色综合国产| 91成年人视频 | 国产99在线免费 | 97超碰国产精品女人人人爽 | 亚洲天堂网在线视频 | 中中文字幕av | 免费一级片观看 | 麻豆免费观看视频 | 麻豆国产在线视频 | 免费成人在线电影 | 免费毛片一区二区三区久久久 | 国产免费黄视频在线观看 | 久久久免费 | 久久国产精品99久久久久久老狼 | 国产精品一区二区久久久 | 操久| 日韩在线小视频 | 亚洲欧美怡红院 | 免费在线一区二区 | 日韩精品一区二区三区高清免费 | 激情五月av | 欧美日韩国产在线 | 精品国产一区二区三区久久久蜜臀 | 国产一区在线不卡 | 高清不卡毛片 | 欧美成人性网 | 日日爱夜夜爱 | 日免费视频 | www天天操 | 国产人成在线视频 | 人人看人人草 | 日韩啪啪小视频 | 一区二区视频电影在线观看 | 91精品国产自产在线观看永久 | 中文字幕色站 | 日日摸日日碰 | 国产麻豆精品在线观看 | 99在线视频观看 | 91成人精品国产刺激国语对白 | 深爱五月激情五月 | 亚洲国产人午在线一二区 | 国产精品午夜在线观看 | 日日干日日操 | 日韩av不卡在线 | 六月丁香婷 | 午夜性生活 | 最近日本mv字幕免费观看 | 国产电影黄色av | 99在线播放 | 九九电影在线 | 91精品一区国产高清在线gif | 国产1区2 | 亚洲干视频在线观看 | 国产成人专区 | 免费观看久久 | 久久不射电影院 | 波多野结衣视频一区二区 | 西西444www大胆高清视频 | 香蕉网站在线观看 | 高清av不卡| 日韩成人高清在线 | 亚洲干视频在线观看 | 一本一道波多野毛片中文在线 | 久草免费福利在线观看 | 国产男女无遮挡猛进猛出在线观看 | 国产一级免费电影 | 天天五月天色 | 国产美女在线观看 | 看片的网址 | 婷婷丁香自拍 | 国产资源精品在线观看 | 久99久在线视频 | 午夜视频免费在线观看 | 久久精品99| 人人干97| 日韩高清在线观看 | 99久久精品免费看国产四区 | 国产精品高 | 久久久久久久久久久久影院 | 婷婷激情5月天 | 国产黑丝一区二区 | 天天操比 | 日韩欧美一区二区三区黑寡妇 | 毛片基地黄久久久久久天堂 | 国产色妞影院wwwxxx | 亚洲狠狠婷婷 | 国产精品免费不卡 | 日韩视频一 | 五月激情丁香图片 | 五月天丁香亚洲 | 久久久久久久久久福利 | 成年人黄色在线观看 | 婷婷久草| 日本资源中文字幕在线 | 安徽妇搡bbbb搡bbbb | www激情网| 亚洲亚洲精品在线观看 | 亚洲精品tv | av资源免费在线观看 | 福利视频导航网址 | 免费在线成人av电影 | 婷婷5月激情5月 | 色99导航 | 深夜视频久久 | 中文字幕在线观看不卡 | 婷婷在线网站 | 91刺激视频 | 欧美精品小视频 | 最近日本韩国中文字幕 | 香蕉视频色 | 国产看片 色| 蜜臀久久99精品久久久久久网站 | 一区二区三区免费在线 | 在线免费亚洲 | 国产精品成人久久久 | 开心色插 | 免费情趣视频 | 午夜精品福利在线 | 日韩欧美一区二区三区视频 | 99re国产视频 | 国产精品美女视频网站 | 久久精品理论 | 亚洲欧美偷拍另类 | 少妇av片 | 91亚洲网站| 久久久18 | 一区二区三区国产精品 | 亚洲精品乱码久久久久久写真 | 91在线观看黄 | 成年人在线观看 | 特级西西人体444是什么意思 | 日本精品一区二区 | 亚洲片在线 | 麻豆视频免费入口 | 国产成人香蕉 | 九九免费在线观看视频 | 久久精品亚洲国产 | 最新日韩在线 | av最新资源| 999久久| 四虎www| 久久夜色电影 | 欧美日韩在线免费视频 | 91在线观看视频网站 | 中文字幕色婷婷在线视频 | 99热精品久久 | 九九av| 亚洲闷骚少妇在线观看网站 | 青青草国产成人99久久 | 国产精品成人久久久久 | 亚洲精品玖玖玖av在线看 | 韩日av一区二区 | 又黄又刺激的视频 | 在线观看av大片 | 国产大陆亚洲精品国产 | 成人永久视频 | 亚洲婷婷丁香 | 久久不射电影网 | 99亚洲视频 | 狠狠狠狠狠狠 | 在线免费视频你懂的 | 久久男人中文字幕资源站 | 99热官网| 九九色在线| 狠狠的日日| 91精品国自产在线 | 亚州精品在线视频 | a视频在线观看 | 欧美激情视频一区二区三区 | 日韩免费在线一区 | 天天曰夜夜爽 | 日韩视频在线不卡 | 超碰在线公开 | 久久亚洲精品电影 | 91在线观看高清 | 黄色国产在线 | 国产精品久久99综合免费观看尤物 | 久久久久免费电影 | 免费观看性生交 | 手机av在线网站 | 天天干夜夜夜操天 | 国产在线中文 | 伊人资源视频在线 | 国产精品久久久久久一二三四五 | 天天色天天操综合网 | 丁香5月婷婷久久 | 96久久欧美麻豆网站 | 综合久久久 | 99精品国产高清在线观看 | 不卡的av中文字幕 | 亚洲少妇天堂 | 最近中文字幕免费观看 | 在线观看国产www | 亚洲欧美成人在线 | 国产精品嫩草影视久久久 | 日韩色爱| 国产成人精品免费在线观看 | 色综合网 | 91黄色免费网站 | 天天干,狠狠干 | 99中文字幕在线观看 | 国产在线高清精品 | 国产高清不卡在线 | 久久免费99精品久久久久久 | 天天操人人干 | 91精品久| 久99精品| 99视频这里只有 | 999久久久欧美日韩黑人 | 欧美日韩二三区 | 麻豆影视在线观看 | 天天草夜夜 | 精品久久久久久久久久久久久久久久久久 | 特级黄色一级 | 午夜精品99久久免费 | 九九综合在线 | 久久a久久 | 96视频在线| 毛片基地黄久久久久久天堂 | www.久久婷婷 | 黄色片网站大全 | 国产91免费在线观看 | 韩日av在线| 毛片网站在线观看 | 久久9999久久免费精品国产 | 在线观看你懂的网址 | 99免费在线| 丁香久久五月 | 日韩精品久久久久久久电影竹菊 | 国产精品一区二区三区四 | 久久精品国产免费看久久精品 | 久久中文精品视频 | 人人爱天天操 | 成人综合婷婷国产精品久久免费 | 色在线免费| 日韩理论电影在线观看 | 国产精品美女视频 | av网站在线观看播放 | 2018亚洲男人天堂 | 免费看黄在线观看 | 最新日韩在线观看视频 | 91精品免费 | 国产精品一区二区中文字幕 | 六月丁香在线视频 | 夜夜天天干 | 亚洲高清在线观看视频 | 99精品国产99久久久久久97 | 黄色特一级片 | 色欲综合视频天天天 | 久久久久麻豆v国产 | 在线看的毛片 | 中文字幕免费高清 | 亚洲精品视频第一页 | 精品在线一区二区三区 | av福利网址导航 | 国产69精品久久久久久 | 玖玖视频在线 | 成人a级黄色片 | 中国一级片免费看 | 国产中文字幕av | 成人午夜免费福利 | 久久免费视频观看 | 国产美女被啪进深处喷白浆视频 | 日本不卡123 | 欧美成人69av | 日日爽夜夜爽 | 91大神dom调教在线观看 | 成人高清在线 | 91视频电影 | 97色资源| 国产一区二区三区网站 | 久久综合精品国产一区二区三区 | 六月激情网 | 鲁一鲁影院 | 日韩免费视频线观看 | 久精品一区| 国产精品久久久久久999 | 伊人网av| 热久久免费视频 | 99久久999久久久精玫瑰 | 久久精品日本啪啪涩涩 | 久久精品亚洲一区二区三区观看模式 | 99精品视频免费看 | 欧美成人播放 | 国产18精品乱码免费看 | 久久精品视频4 | 激情欧美一区二区免费视频 | 毛片1000部免费看 | 日本mv大片欧洲mv大片 | 日韩成人黄色av | 日韩av在线资源 | 国产一区二区不卡视频 | 日日干美女| 欧美成人精品三级在线观看播放 | 色综合久久88 | 99re中文字幕 | 免费看的黄网站软件 | 99热99 | 久久精品高清视频 | 狠狠干 狠狠操 | 久久亚洲在线 | 激情视频一区 | 国产一级免费播放 | 久久婷婷精品视频 | 色妞久久福利网 | 婷婷爱五月天 | 天天激情在线 | 亚洲精品在线免费 | 免费中午字幕无吗 | 亚洲老妇xxxxxx | 91完整版在线观看 | 日韩激情在线 | 久久66热这里只有精品 | 福利视频导航网址 | 中文一二区 | 国产高清视频在线 | 亚洲免费av观看 | 久久久亚洲精华液 | 日韩精品播放 | 成人超碰97 | 成人在线视频免费看 | 在线电影日韩 | 六月激情 | 久久免费福利视频 | 色天天综合网 | 国产黄色片一级三级 | 日本精品在线 | 人人澡人人添人人爽一区二区 | 亚洲精品黄 | 精品免费观看视频 | 欧美精品你懂的 | 亚洲 欧美 变态 国产 另类 | 国产精品久久久久久久久软件 | 国产色黄网站 | 天天摸天天干天天操天天射 | 91人人爱 | 国产免费精彩视频 | 亚洲久草在线视频 | 国产123区在线观看 国产精品麻豆91 | 午夜精品久久久久久久99 | 亚洲国产精品va在线看黑人动漫 | 亚洲精品一区二区三区四区高清 | 久久午夜网 | 久草视频精品 | 日韩欧美在线视频一区二区三区 | 久久精品视频免费观看 | 人人舔人人插 | 国产色 在线 | 中文字幕在线字幕中文 | 狠狠撸电影| 久久久久久久久久影院 | 日日操狠狠干 | 国产精品免费久久久久影院仙踪林 | 天天曰夜夜爽 | 亚洲天天做 | 亚洲人av免费网站 | 亚洲黄色一级视频 | 成人免费观看视频大全 | 亚洲永久精品在线观看 | 免费观看高清 | 久久精品在线免费观看 | 日韩精品一区二区三区视频播放 | 国产成人av一区二区三区在线观看 | 久久人人爽人人爽 | 精品久久久久久久久久久院品网 | 99久久99视频只有精品 | 国产精品美女久久久久久久久 | 少妇性色午夜淫片aaaze | 国产一区欧美在线 | 婷婷av综合 | 综合久久久久久久 | 97在线视频免费播放 | 婷婷四房综合激情五月 | 国产午夜三级一区二区三桃花影视 | 亚洲免费av网站 | 色九九视频 | 国产第一页在线播放 | 波多野结衣电影久久 | 亚洲色视频| 免费毛片一区二区三区久久久 | 又黄又刺激的网站 | 69av免费视频 | www.xxx.性狂虐 | 色多多视频在线观看 | 懂色av一区二区三区蜜臀 | 天天干天天在线 | 亚洲欧洲精品在线 | 天天操天天操天天操天天 | 伊人婷婷久久 | 91精品啪在线观看国产81旧版 | 国产在线播放一区二区 | 久久色在线播放 | 亚洲精品视频在线观看免费视频 | 亚洲欧洲精品一区 | 亚洲视频一级 | 欧美福利在线播放 | 日韩美视频 | 日韩二区在线观看 | 久草91视频| www91在线观看 | 亚洲国产网站 | 国产精品青青 | 色播99 | 免费看的黄色录像 | 精品国产免费久久 | 久久久久久久久精 | 亚洲激情在线播放 | 国产色综合天天综合网 | 日韩大片在线免费观看 | 日韩婷婷 | 黄色成人免费电影 | 国产做爰视频 | 亚洲专区在线视频 | 在线а√天堂中文官网 | 亚洲精品乱码久久久久久 | 亚洲视频精选 | 91豆麻精品91久久久久久 | 久久久蜜桃一区二区 | 亚洲成a人片77777kkkk1在线观看 | 97av色| 免费观看午夜视频 | 日本一区二区三区免费看 | av成人免费在线 | 一区二区三区四区影院 | 亚洲一级性 | 国产小视频91 | 欧美成人xxxxxxxx | 99久久久久免费精品国产 | 亚洲精品久久久蜜桃直播 | 国产精品免费视频久久久 | 视频在线精品 | 日韩精品一区二区三区中文字幕 | 日免费视频 | 四虎视频 | 日韩av网页 | 深爱激情五月婷婷 | 亚洲欧美日韩精品一区二区 | 97网站| 久久久久97国产 | 青春草免费视频 | 韩国av免费在线 | 97在线观看免费 | 久久久久久美女 | 中文字幕你懂的 | 日韩精选在线观看 | 久久色网站 | 亚洲精品久久激情国产片 | 国产高清日韩欧美 | 日日精品 | 国产不卡在线观看 | 992tv成人免费看片 | 黄色大片日本 | 亚洲在线日韩 | 91看片在线观看 | 超碰在线1| 最新国产在线 | 成人一区在线观看 | 国产精品视频内 | 日韩av在线看 | 国产破处在线播放 | 久99视频| 97国产在线播放 | 在线视频精品播放 | 亚洲国产精品成人综合 | 亚洲成年人在线播放 | 久久精品久久久久电影 | 99久久99久久精品 | 免费在线观看日韩欧美 | 亚洲美女视频网 | 日韩三级视频在线观看 | 国产一线二线三线在线观看 | 亚洲一级影院 | 成年人av在线播放 | 激情网站网址 | 手机av电影在线 | 久久综合视频网 | 成人av一区二区三区 | 97**国产露脸精品国产 | 97超碰免费 | 狠狠干天天干 | 久久久亚洲麻豆日韩精品一区三区 | 亚洲成av人片在线观看香蕉 | 欧美日韩精品在线播放 | 免费视频一级片 | 婷婷激情综合五月天 | 成人毛片在线观看 | 天天射天天干天天插 | 在线观看黄av | 91精品国产高清自在线观看 | 久久午夜国产精品 | 精品日韩中文字幕 | 亚洲一级片在线观看 | 欧美激情视频免费看 | 国产欧美精品一区二区三区 | 国产一区免费在线观看 | 美女在线观看网站 | 97超碰网 | 伊人色综合网 | 粉嫩一区二区三区粉嫩91 | 色开心| 手机成人av | av电影免费看 | 波多野结衣在线视频免费观看 | 国产一区91 | 久久久久一区二区三区 | 成人一区二区在线观看 | 四虎在线观看精品视频 | 精品 一区 在线 | 国产精品视频在线观看 | 中文字幕日韩av | 97av视频在线 | 人人精久| 国产精品美女久久久久久久久久久 | av亚洲产国偷v产偷v自拍小说 | 四虎免费在线观看视频 | 精品一区二区在线观看 | 色无五月| 久久国产成人午夜av影院宅 | 午夜视频免费 | 国产在线观看,日本 | 成人在线观看资源 | 久草在线欧美 | 手机av资源 | 久久免费公开视频 | 久久国产麻豆 | 蜜臀久久99精品久久久无需会员 | 欧美色插 | 久草免费资源 | 国内精品久久久久影院男同志 | 最新超碰 | 午夜三级理论 | 中文字幕在线日亚洲9 | 精品久久精品久久 | 亚洲精品国偷拍自产在线观看蜜桃 | se婷婷| 久久艹国产视频 | 天天干天天做 | 国产精品小视频网站 | 久久久久久蜜av免费网站 | 亚洲91中文字幕无线码三区 | 99精品偷拍视频一区二区三区 | 久草视频在| 国产91综合一区在线观看 | 欧美久久久久久久久久久久久 | 天天射天天干天天操 | 精品国产乱码久久久久久浪潮 | 婷婷色在线观看 | 成 人 黄 色 视频播放1 | 岛国av在线免费 | 97色综合 | 免费试看一区 | 欧美激情第八页 | 国产精品美女视频 | av成人在线观看 | 91av视屏 | 亚洲成人av一区二区 | 免费看一级特黄a大片 | 久久男人中文字幕资源站 | 天干啦夜天干天干在线线 | 中文字幕首页 | 国产日韩精品一区二区在线观看播放 | 国产综合在线视频 | 69久久久| 久久一级电影 | 又黄又爽又无遮挡免费的网站 | 午夜精品婷婷 | 欧美日韩精品免费观看 | 国产精久久久久久妇女av | 狠狠的干狠狠的操 | 婷婷国产v亚洲v欧美久久 | 国产精品少妇 | 久久免费精品视频 | 免费在线激情电影 | 亚洲国产精品日韩 | 又黄又爽的免费高潮视频 | 黄色免费看片网站 | 99久高清在线观看视频99精品热在线观看视频 | 在线免费av观看 | 中文字幕影视 | 天天干天天操天天入 | 精品免费观看视频 | 日日噜噜噜噜夜夜爽亚洲精品 | 天天干天天射天天操 | 亚洲五月 | 日韩色视频在线观看 | 在线黄频 | 西西大胆啪啪 | 天天曰天天 | 国产一卡久久电影永久 | 看av在线| 日韩在线在线 | 国产福利在线不卡 | 97超在线视频 | 亚洲黄色在线免费观看 | 国产精品免费观看久久 | 国产成人精品久久久 | 五月开心婷婷网 | 中文字幕久久精品亚洲乱码 | 精品国产乱码久久久久 | 欧美亚洲成人免费 | 四虎永久免费在线观看 | 毛片精品免费在线观看 | 亚洲欧美激情精品一区二区 | 国产成人精品一区二区三区 | 久久综合久久综合这里只有精品 | 欧美日韩高清一区二区 国产亚洲免费看 | 国产黄色理论片 | 国产在线国产 | 97人人澡人人爽人人模亚洲 | 婷婷在线免费观看 | 日韩经典一区二区三区 | 免费看一级一片 | 狂野欧美激情性xxxx | 亚洲乱码中文字幕综合 | 天天射天天艹 | av在线h| 99久久er热在这里只有精品66 | 欧美色久 | 国产日韩精品一区二区三区在线 | 久久99精品久久久久久清纯直播 | 国产亚洲精品美女 | 久久av免费电影 | 国产精品a成v人在线播放 | 国产日韩视频在线观看 | 不卡的av在线播放 | 久久久久电影 | 色瓜 | 伊人国产在线观看 | 中文字幕第一页在线视频 | 国语久久| 色婷婷久久一区二区 | 成人免费视频视频在线观看 免费 | 色吊丝在线永久观看最新版本 | 色婷婷免费| 久久爱992xxoo | 91资源在线 | 手机在线永久免费观看av片 | 免费在线观看国产精品 | 成人国产精品久久久久久亚洲 | 欧美日韩有码 | 69精品视频在线观看 | 久久99久久99精品免观看软件 | 天天拍天天干 | 99久久超碰中文字幕伊人 | 狠狠色丁香久久婷婷综合_中 | 一区二区免费不卡在线 | www国产精品com | 免费av片在线 | 偷拍福利视频一区二区三区 | 天堂av在线中文在线 | 免费看精品久久片 | 波多野结衣在线视频免费观看 | 亚洲精品一区二区精华 | 成人黄色片免费看 | 天天操天天色天天射 | 国产成人中文字幕 | 91精品一 | 国内精品久久久久久中文字幕 | 天天色综合三 | 日韩免费观看视频 | 国产精品不卡在线观看 | 中文字幕乱码一区二区 | 五月天久久精品 | 人人爱天天操 | 麻豆视频在线看 | www.超碰 | 成人网大片 | 亚洲精品乱码久久久久久高潮 | 精品久久久久久久久久 | 超碰成人免费电影 | 久艹在线观看视频 | 日韩理论电影在线 | 中文字幕在线看视频 | 天天干天天天天 | 亚洲黑丝少妇 | 国产精品1区 | 国产中文字幕免费 | 日本在线观看视频一区 | 操老逼免费视频 | 看毛片网站 | 色噜噜日韩精品欧美一区二区 | 91九色视频网站 | 六月丁香婷婷久久 | 97超碰人人 | 亚洲欧洲在线视频 | 久久久福利视频 | av五月婷婷| 日韩精选在线 | 天天干天天综合 | 国产精品视频在线观看 | 久久成人在线视频 | 狠狠操狠狠干天天操 | 特级大胆西西4444www |