91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

OkHttp如何實現透明壓縮

發布時間:2022-02-24 11:18:19 來源:億速云 閱讀:146 作者:iii 欄目:開發技術

這篇文章主要介紹了OkHttp如何實現透明壓縮的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇OkHttp如何實現透明壓縮文章都會有所收獲,下面我們一起來看看吧。

什么叫透明壓縮呢?OkHttps 在發送請求的時候,會自動加入 gzip 請求頭Accept-Encoding:gzip。所以,當返回的數據帶有 gzip 響應頭時Content-Encoding=gzip,OkHttps 會自動幫我們解壓數據。(Accept-EncodingContent-Encoding是一對請求頭,分別對應著請求和返回)

為什么要進行壓縮呢?因為它能大幅減少傳輸的容量。像一些 CPU 資源占用不高的服務,比如 Kafka ,我們就可以開啟 gzip 壓縮,加快信息的流轉。

這個壓縮比有多高呢?對于普通的xml或者json,數據可以由9MB壓縮到350KB左右,壓縮比足足達到了26

它讓系統性能飛起來

SpringCloud微服務體系,現在有非常多的公司在用。即使是一些傳統企業,一些大數據量的toB企業,也想嘗一嘗螃蟹。

對于一個簡單的 SpringBoot 服務,我們只需要在 yml 文件中配置上相應的壓縮就可以了。這樣,我們就打通了瀏覽器到 Web 服務的這一環。這種壓縮方式,對于大數據量的服務來說,是救命式的!

具體配置如下。

server:
  port: 8082
  compression:
    enabled: true
    min-response-size: 1024
    mime-types: ["text/html","text/xml","application/xml","application/json","application/octet-stream"]

它所對應的 Spring 配置類是org.springframework.boot.web.server.Compression

但是不要高興太早。由于是分布式環境,這里面調用鏈就會長一些。即使是在內網,動輒十幾MB的網絡傳輸,也會耗費可觀的時間。

一個請求從瀏覽器到達真正的服務節點,可能要經過很多環節。

  • nginx轉發請求到微服務網關zuul

  • zuul轉發到具體的微服務A

  • 微服務A通過Feign接口調用微服務B

如果我們的數據,大多數是由微服務B提供的,那么上面的任何一個環節傳輸效率慢,都會影響請求的性能。

所以,我們需要開啟 Feign 接口的 gzip 壓縮。使用 OkHttps 的透明代理是最簡單的方式。

首先,在項目中引入 feign 的 jar 包。

<dependency>
            <groupId>io.github.openfeign</groupId>
            <artifactId>feign-okhttp</artifactId>
</dependency>

其次,在 yml 文件中啟用 OkHttps 作為 feign 的客戶端請求工具包。穩妥起見,我們同時屏蔽了 httpclient ,這個東西太重太老了。

feign:
  httpclient:
    enabled: false
  okhttp:
    enabled: true

到此為止,我們就可以享受 OkHttps 的透明代理帶來的便捷性了。

假如你的應用數據包大,調用鏈長,這種方式甚至會給你的服務帶來數秒的性能提升。 xjjdog 就曾經靠調整幾個參數,就讓一個蝸牛系統飛了起來。大家驚呼:原來B端也可以C一下。

OkHttp是如何實現透明壓縮的?

OkHttps 對于透明壓縮的處理,是通過攔截器來做的。具體的類,就是okhttp3.internal.http.BridgeInterceptor

具體代碼如下,當判斷沒有Accept-Encoding頭的時候,就自行加入一個。

// If we add an "Accept-Encoding: gzip" header field we're responsible for also decompressing
// the transfer stream.
boolean transparentGzip = false;
if (userRequest.header("Accept-Encoding") == null && userRequest.header("Range") == null) {
  transparentGzip = true;
  requestBuilder.header("Accept-Encoding", "gzip");
}

最關鍵的代碼在下面。

if (transparentGzip
    && "gzip".equalsIgnoreCase(networkResponse.header("Content-Encoding"))
    && HttpHeaders.hasBody(networkResponse)) {
  GzipSource responseBody = new GzipSource(networkResponse.body().source());
  Headers strippedHeaders = networkResponse.headers().newBuilder()
      .removeAll("Content-Encoding")
      .removeAll("Content-Length")
      .build();
  responseBuilder.headers(strippedHeaders);
  String contentType = networkResponse.header("Content-Type");
  responseBuilder.body(new RealResponseBody(contentType, -1L, Okio.buffer(responseBody)));
}

可以看到if語句里,有三個條件。

  • 程序沒有設置Accept-Encoding,啟用了透明壓縮

  • 服務端有Content-Encoding頭,并啟用了gzip壓縮

  • 有數據包

只有同時滿足這三個條件,OkHttps 的透明壓縮才會起作用,幫我們自動解壓。

它挖的坑有點深

可惜的是,上面的關鍵代碼,只有if,沒有else,也就是當其中的任何一個條件不滿足,后端的數據包將原封不動的返回。

2、3兩個條件是沒有什么問題的,原樣返回后端數據并沒有什么損害,問題就出在第一個條件里。

如果你在代碼中,使用了下面的代碼:

Request.Builder builder = chain.request()
                .newBuilder()
                .addHeader("Accept", "application/json")
                .addHeader("Accept-Encoding", "gzip");

也就是手動設置了Accept-Encoding頭信息。這很常見,因為這體現了程序員思維的嚴謹。

正是這種嚴謹,造成了問題。

假如你的后端應用剛開始是沒有開啟gzip壓縮的,這時候兩者相安無事;但如果你的后端應用突然有一天開啟了gzip壓縮,你的這段代碼將全部over。

原因就是,服務端gzip數據包會原樣返回,你需要手動處理gzip數據包。

所以,不加是好事,加了反而會壞事,除非你想自己處理gzip數據。

由于OkHttpAndroid上應用也非常廣泛,如果你不知道這個細節,造成的后果就是災難性的。客戶端更新慢,只能老老實實回退服務端了。

關于“OkHttp如何實現透明壓縮”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“OkHttp如何實現透明壓縮”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

通山县| 安多县| 增城市| 华坪县| 叶城县| 巴楚县| 衡阳县| 滁州市| 台南县| 井研县| 门头沟区| 乃东县| 盐山县| 凤冈县| 饶河县| 桑日县| 石景山区| 宜兴市| 揭西县| 富川| 包头市| 抚远县| 乌拉特前旗| 南靖县| 石棉县| 普宁市| 洛宁县| 泰宁县| 岱山县| 承德市| 南投县| 怀集县| 光泽县| 城固县| 宣恩县| 新竹市| 安福县| 民勤县| 遂昌县| 陆良县| 隆林|