圖片優化——質量與性能的博弈

就圖片這塊來說, 與其說是在做優化, 不如說實在做權衡。因爲我們要做的事情, 就是去壓縮圖片的體積。但是這個操作, 是以犧牲一部分成像質量爲代價的。因此我們的主要任務, 是儘可能的去尋求一個質量與性能之間的平衡點。

不同業務場景下的圖片方案選型

時下應用較爲廣泛的 Web 圖片格式有 JPEG/JPG、PNG、WebP、Base64、SVG 等,這些格式都是很有故事的,值得我們好好研究一把。此外,老生常談的雪碧圖(CSS Sprites)至今也仍在一線的前端應用中發光發熱,我們也會有所提及。

前置知識:二進制位數與色彩的關係

在計算機中,像素用二進制數來表示。不同的圖片格式中像素與二進制位數之間的對應關係是不同的。一個像素對應的二進制位數越多,它可以表示的顏色種類就越多,成像效果也就越細膩,文件體積相應也會越大。

一個二進制位表示兩種顏色(0|1 對應黑|白),如果一種圖片格式對應的二進制位數有 n 個,那麼它就可以呈現 2^n 種顏色。

JPEG/JPG

關鍵字:有損壓縮、體積小、加載快、不支持透明

JPG 的優點

JPG 最大的特點是有損壓縮。這種高效的壓縮算法使它成爲了一種非常輕巧的圖片格式。另一方面,即使被稱爲“有損”壓縮,JPG的壓縮方式仍然是一種高質量的壓縮方式:當我們把圖片體積壓縮至原有體積的 50% 以下時,JPG 仍然可以保持住 60% 的品質。此外,JPG 格式以 24 位存儲單個圖,可以呈現多達 1600 萬種顏色,足以應對大多數場景下對色彩的要求,這一點決定了它壓縮前後的質量損耗並不容易被我們人類的肉眼所察覺——前提是你用對了業務場景。

使用場景

JPG 適用於呈現色彩豐富的圖片,在我們日常開發中,JPG 圖片經常作爲大的背景圖、輪播圖或 Banner 圖出現。

兩大電商網站對大圖的處理,是 JPG 圖片應用場景的最佳寫照:

打開淘寶首頁,我們可以發現頁面中最醒目、最龐大的圖片,一定是以 .jpg 爲後綴的:

使用 JPG 呈現大圖,既可以保住圖片的質量,又不會帶來令人頭疼的圖片體積,是當下比較推崇的一種方案。

JPG 的缺陷

有損壓縮在上文所展示的輪播圖上確實很難露出馬腳,但當它處理矢量圖形和 Logo 等線條感較強、顏色對比強烈的圖像時,人爲壓縮導致的圖片模糊會相當明顯。

此外,JPEG 圖像不支持透明度處理,透明圖片需要召喚 PNG 來呈現。

PNG-8 與 PNG-24

關鍵字:無損壓縮、質量高、體積大、支持透明

PNG 的優點

PNG(可移植網絡圖形格式)是一種無損壓縮的高保真的圖片格式。8 和 24,這裏都是二進制數的位數。按照我們前置知識裏提到的對應關係,8 位的 PNG 最多支持 256 種顏色,而 24 位的可以呈現約 1600 萬種顏色。

PNG 圖片具有比 JPG 更強的色彩表現力,對線條的處理更加細膩,對透明度有良好的支持。它彌補了上文我們提到的 JPG 的侷限性,唯一的 BUG 就是體積太大

PNG-8 與 PNG-24 的選擇題

什麼時候用 PNG-8,什麼時候用 PNG-24,這是一個問題。

理論上來說,當你追求最佳的顯示效果、並且不在意文件體積大小時,是推薦使用 PNG-24 的。

但實踐當中,爲了規避體積的問題,我們一般不用PNG去處理較複雜的圖像。當我們遇到適合 PNG 的場景時,也會優先選擇更爲小巧的 PNG-8。

如何確定一張圖片是該用 PNG-8 還是 PNG-24 去呈現呢?好的做法是把圖片先按照這兩種格式分別輸出,看 PNG-8 輸出的結果是否會帶來肉眼可見的質量損耗,並且確認這種損耗是否在我們(尤其是你的 UI 設計師)可接受的範圍內,基於對比的結果去做判斷。

應用場景

前面我們提到,複雜的、色彩層次豐富的圖片,用 PNG 來處理的話,成本會比較高,我們一般會交給 JPG 去存儲。

考慮到 PNG 在處理線條和顏色對比度方面的優勢,我們主要用它來呈現小的 Logo、顏色簡單且對比強烈的圖片或背景等。

此時我們再次把目光轉向性能方面堪稱業界楷模的淘寶首頁,我們會發現它頁面上的 Logo,無論大小,還真的都是 PNG 格式:

SVG

關鍵字:文本文件、體積小、不失真、兼容性好

SVG(可縮放矢量圖形)是一種基於 XML 語法的圖像格式。它和本文提及的其它圖片種類有着本質的不同:SVG 對圖像的處理不是基於像素點,而是是基於對圖像的形狀描述。

SVG 的特性

和性能關係最密切的一點就是:SVG 與 PNG 和 JPG 相比,文件體積更小,可壓縮性更強

當然,作爲矢量圖,它最顯著的優勢還是在於圖片可無限放大而不失真這一點上。這使得 SVG 即使是被放到視網膜屏幕上,也可以一如既往地展現出較好的成像品質——1 張 SVG 足以適配 n 種分辨率。

此外,SVG 是文本文件。我們既可以像寫代碼一樣定義 SVG,把它寫在 HTML 裏、成爲 DOM 的一部分,也可以把對圖形的描述寫入以 .svg 爲後綴的獨立文件(SVG 文件在使用上與普通圖片文件無異)。這使得 SVG 文件可以被非常多的工具讀取和修改,具有較強的靈活性

SVG 的侷限性主要有兩個方面,一方面是它的渲染成本比較高,這點對性能來說是很不利的。另一方面,SVG 存在着其它圖片格式所沒有的學習成本(它是可編程的)。

SVG 的使用方式與應用場景

SVG 是文本文件,我們既可以像寫代碼一樣定義 SVG,把它寫在 HTML 裏、成爲 DOM 的一部分,也可以把對圖形的描述寫入以 .svg 爲後綴的獨立文件(SVG 文件在使用上與普通圖片文件無異)。

  • 將 SVG 寫入 HTML:

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title></title>
    </head>
    <body>
        <svg xmlns="http://www.w3.org/2000/svg"   width="200" height="200">
            <circle cx="50" cy="50" r="50" />
        </svg>
    </body>
    </html>

     

  • 將 SVG 寫入獨立文件後引入 HTML:
<img src="文件名.svg" alt="">

在實際開發中,我們更多用到的是後者。很多情況下設計師會給到我們 SVG 文件,就算沒有設計師,我們還有非常好用的 在線矢量圖形庫。對於矢量圖,我們無須深究過多,只需要對其核心特性有所掌握、日後在應用時做到有跡可循即可。

Base64

關鍵字:文本文件、依賴編碼、小圖標解決方案

Base64 並非一種圖片格式,而是一種編碼方式。Base64 和雪碧圖一樣,是作爲小圖標解決方案而存在的。在瞭解 Base64 之前,我們先來了解一下雪碧圖。

前置知識:最經典的小圖標解決方案——雪碧圖(CSS Sprites)

雪碧圖、CSS 精靈、CSS Sprites、圖像精靈,說的都是這個東西——一種將小圖標和背景圖像合併到一張圖片上,然後利用 CSS 的背景定位來顯示其中的每一部分的技術。

MDN 對雪碧圖的解釋已經非常到位:

圖像精靈(sprite,意爲精靈),被運用於衆多使用大量小圖標的網頁應用之上。它可取圖像的一部分來使用,使得使用一個圖像文件替代多個小文件成爲可能。相較於一個小圖標一個圖像文件,單獨一張圖片所需的 HTTP 請求更少,對內存和帶寬更加友好。

理解 Base64

通過我們上文的演示,大家不難看出,每次加載圖片,都是需要單獨向服務器請求這個圖片對應的資源的——這也就意味着一次 HTTP 請求的開銷。

Base64 是一種用於傳輸 8Bit 字節碼的編碼方式,通過對圖片進行 Base64 編碼,我們可以直接將編碼結果寫入 HTML 或者寫入 CSS,從而減少 HTTP 請求的次數。

我們來一起看一個實例,現在我有這麼一個小小的放大鏡 Logo:

它對應的鏈接如下:

https://user-gold-cdn.xitu.io/2018/9/15/165db7e94699824b?w=22&h=22&f=png&s=3680

按照一貫的思路,我們加載圖片需要把圖片鏈接寫入 img 標籤:

<img src="https://user-gold-cdn.xitu.io/2018/9/15/165db7e94699824b?w=22&h=22&f=png&s=3680">

瀏覽器就會針對我們的圖片鏈接去發起一個資源請求。

但是如果我們對這個圖片進行 Base64 編碼,我們會得到一個這樣的字符串:



字符串比較長,我們可以直接用這個字符串替換掉上文中的鏈接地址。你會發現瀏覽器原來是可以理解這個字符串的,它自動就將這個字符串解碼爲了一個圖片,而不需再去發送 HTTP 請求。

Base64 的應用場景

一般Base64 碼出現的地方它對應的圖片往往是非常小的 Logo

既然 Base64 這麼棒,我們何不把大圖也換成 Base64 呢?

這是因爲,Base64 編碼後,圖片大小會膨脹爲原文件的 4/3(這是由 Base64 的編碼原理決定的)。如果我們把大圖也編碼到 HTML 或 CSS 文件中,後者的體積會明顯增加,即便我們減少了 HTTP 請求,也無法彌補這龐大的體積帶來的性能開銷,得不償失。
在傳輸非常小的圖片的時候,Base64 帶來的文件體積膨脹、以及瀏覽器解析 Base64 的時間開銷,與它節省掉的 HTTP 請求開銷相比,可以忽略不計,這時候才能真正體現出它在性能方面的優勢。

因此,Base64 並非萬全之策,我們往往在一張圖片滿足以下條件時會對它應用 Base64 編碼:

  • 圖片的實際尺寸很小(大家可以觀察一下掘金頁面的 Base64 圖,幾乎沒有超過 2kb 的)
  • 圖片無法以雪碧圖的形式與其它小圖結合(合成雪碧圖仍是主要的減少 HTTP 請求的途徑,Base64 是雪碧圖的補充)
  • 圖片的更新頻率非常低(不需我們重複編碼和修改文件內容,維護成本較低)

Base64 編碼工具推薦

這裏最推薦的是利用 webpack 來進行 Base64 的編碼——webpack 的 url-loader 非常聰明,它除了具備基本的 Base64 轉碼能力,還可以結合文件大小,幫我們判斷圖片是否有必要進行 Base64 編碼。

除此之外,市面上免費的 Base64 編解碼工具種類是非常多樣化的,有很多網站都提供在線編解碼的服務,大家選取自己認爲順手的工具就好。

WebP

關鍵字:年輕的全能型選手

WebP 是今天在座各類圖片格式中最年輕的一位,它於 2010 年被提出, 是 Google 專爲 Web 開發的一種旨在加快圖片加載速度的圖片格式,它支持有損壓縮和無損壓縮。

WebP 的優點

WebP 像 JPEG 一樣對細節豐富的圖片信手拈來,像 PNG 一樣支持透明,像 GIF 一樣可以顯示動態圖片——它集多種圖片文件格式的優點於一身。
WebP 的官方介紹對這一點有着更權威的闡述:

與 PNG 相比,WebP 無損圖像的尺寸縮小了 26%。在等效的 SSIM 質量指數下,WebP 有損圖像比同類 JPEG 圖像小 25-34%。 無損 WebP 支持透明度(也稱爲 alpha 通道),僅需 22% 的額外字節。對於有損 RGB 壓縮可接受的情況,有損 WebP 也支持透明度,與 PNG 相比,通常提供 3 倍的文件大小。

我們開篇提到,圖片優化是質量與性能的博弈,從這個角度看,WebP 無疑是真正的贏家。

WebP 的侷限性

WebP 縱有千般好,但它畢竟太年輕。我們知道,任何新生事物,都逃不開兼容性的大坑。現在是 2018 年 9 月,WebP 的支持情況是這樣的:

坦白地說,雖然沒有特別慘(畢竟還有親爹 Chrome 在撐腰),但也足夠讓人望而卻步了。

此外,WebP 還會增加服務器的負擔——和編碼 JPG 文件相比,編碼同樣質量的 WebP 文件會佔用更多的計算資源。

 

WebP 的應用場景

現在限制我們使用 WebP 的最大問題不是“這個圖片是否適合用 WebP 呈現”的問題,而是“瀏覽器是否允許 WebP”的問題,即我們上文談到的兼容性問題。具體來說,一旦我們選擇了 WebP,就要考慮在 Safari 等瀏覽器下它無法顯示的問題,也就是說我們需要準備 PlanB,準備降級方案。

目前真正把 WebP 格式落地到網頁中的網站並不是很多,這其中淘寶首頁對 WebP 兼容性問題的處理方式就非常有趣。我們可以打開 Chrome 的開發者工具搜索其源碼裏的 WebP 關鍵字:

我們會發現檢索結果還是挺多的(單就圖示的加載結果來看,足足有 200 多條),下面大家注意一下這些 WebP 圖片的鏈接地址(以其中一個爲例):

<img src="//img.alicdn.com/tps/i4/TB1CKSgIpXXXXccXXXX07tlTXXX-200-200.png_60x60.jpg_.webp" alt="手機app - 聚划算" class="app-icon">

.webp 前面,還跟了一個 .jpg 後綴!

我們現在先大膽地猜測,這個圖片應該至少存在 jpg 和 webp 兩種格式,程序會根據瀏覽器的型號、以及該型號是否支持 WebP 這些信息來決定當前瀏覽器顯示的是 .webp 後綴還是 .jpg 後綴。帶着這個預判,我們打開並不支持 WebP 格式的 Safari 來進入同樣的頁面,再次搜索 WebP 關鍵字:

Safari 提示我們找不到,這也是情理之中。我們定位到剛剛示例的 WebP 圖片所在的元素,查看一下它在 Safari 裏的圖片鏈接:

<img src="//img.alicdn.com/tps/i4/TB1CKSgIpXXXXccXXXX07tlTXXX-200-200.png_60x60.jpg" alt="手機app - 聚划算" class="app-icon">

我們看到同樣的一張圖片,在 Safari 中的後綴從 .webp 變成了 .jpg!看來果然如此——站點確實是先進行了兼容性的預判,在瀏覽器環境支持 WebP 的情況下,優先使用 WebP 格式,否則就把圖片降級爲 JPG 格式(本質是對圖片的鏈接地址作簡單的字符串切割)。

此外,還有另一個維護性更強、更加靈活的方案——把判斷工作交給後端,由服務器根據 HTTP 請求頭部的 Accept 字段來決定返回什麼格式的圖片。當 Accept 字段包含 image/webp 時,就返回 WebP 格式的圖片,否則返回原圖。這種做法的好處是,當瀏覽器對 WebP 格式圖片的兼容支持發生改變時,我們也不用再去更新自己的兼容判定代碼,只需要服務端像往常一樣對 Accept 字段進行檢查即可。

由此也可以看出,我們 WebP 格式的侷限性確實比較明顯,如果決定使用 WebP,兼容性處理是必不可少的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章