从图片优化说起

图片是大部分网页的重要组成部分,一般情况下,我们不会太关注这方面的问题,需要显示图片直接一个 img 标签搞定。

但实际上,无论是对于提高加载速度,还是对于优化用户体验,优化图片都是一个重要的手段。

图片优化分成两个方面:

第一,图片压缩。在保证视觉效果的情况下,减少图片的体积。这个很有效,1M 和 100K 的图片,肉眼看起来几乎差不多,但却省了 90% 的流量,大大提高了加载速度。

第二,响应式图片。根据客户端的情况,选择最合适的图片返回给用户。用户是一个 500px 的设备,那么返回 1000px 的图给他就是浪费。

我们先来看图片压缩。

图片压缩

压缩的第一步是筛选出需要压缩的图片。如果图片本身就已经足够小了,那么再压缩的意义就不大。

我一般使用如下的脚本筛选项目中需要压缩的图片。脚本会列出所有的图片并根据尺寸降序排列。

# fd 是现代化的 find
# bat 是现代化的 cat
fd -e png -e jpeg -e jpg -e svg |\
xargs ls -l |\
sort -nk5 -r |\
awk '{print $9,$5}' |\
numfmt --field=2 --to=iec |\
column -t | bat

筛选出需要压缩的图片以后,接下来就是压缩、比对、调整参数。图片压缩的工具实在是太多了,Google image compression tool 选择会多得你眼花缭乱。

这里顺口提一下 Google 出品的 squoosh 在线图片压缩服务,看起来不错,虽然我没怎么用过。

这里我选择使用 imagemin,相比于一些在线工具或者 App,自己写脚本更灵活一些。

程序很简单,分别针对 JPG、PNG、SVG 加载相应的插件就好。

const imagemin = require('imagemin')
const imageminMozjpeg = require('imagemin-mozjpeg')
const imageminPngquant = require('imagemin-pngquant')
const imageminSvgo = require('imagemin-svgo')

;(async () => {
  const files = await imagemin(process.argv.slice(2), {
    destination: 'dist',
    plugins: [
      imageminMozjpeg({
        quality: 70,
      }),
      imageminPngquant({
        quality: [0.65, 0.8],
      }),
      imageminSvgo({
        plugins: [{ removeViewBox: false }],
      }),
    ],
  })
})()

注意,quality 参数需要自己测试去确定,怎样在质量和尺寸中权衡,每个团队有自己的标准。

Progressive VS Baseline

JPEG 根据显示方式的不同,分为两种。Progressive JPEG 会先加载模糊的整张图片,然后变的越来越清晰。

而 Baseline JPEG 会先清晰地加载图片的一部分,然后慢慢显示剩余的部分。

从视觉效果来说,Progressive JPEG 自然更好一些。但它也有一些缺点,比如它的解码速度比 Baseline JPEG 要慢,占用的 CPU 时间更多。

如果是桌面浏览器,这点性能问题自然无所谓,但是如果是移动端,就不得不考虑。工程本来就是权衡的艺术。

默认情况下,MozJPEG 生成的是 Progressive JPEG,可以通过 选项 调整。

WebP

WebP 是谷歌新提出的一个图片格式,拥有质量好尺寸小的特点。在客户端支持的情况下,我们应该尽可能地使用 WebP 格式。

有很多工具可以将 JPG/PNG 转换成 WebP,这里还是使用 imagemin 为例。

const imageminWebp = require('imagemin-webp')

const webps = await imagemin(images, {
  destination: 'dist',
  plugins: [
    imageminWebp({
      quality: 80,
    }),
  ],
})

oimg

oimg 是我在 imagemin 的基础上封装的一个命令行小工具,毕竟压缩图片是经常要做的事情,不能每次都等到需要的时候再去写脚本。

oimg 使的流程是这样的:

  • 首先,我们找到尺寸比较大的需要压缩的图片
  • 然后,使用 oimg 压缩
  • 最后,肉眼对比一下原图片和压缩图片,如果没有问题,替换就好
  • 如果效果不满意,调整参数,再压缩

这个过程没法完全自动化,因为压缩过后的图片究竟在视觉上能不能替换原图,这个过程需要人来判别,全部交给机器是不太放心的。毕竟只有在保证质量的情况下减小体积才有意义。

oimg 的输出如下,可以很方便地看出压缩的效果如何。

响应式图片

图片压缩的问题解决完了,现在我们来看看响应式图片。

所谓响应式图片,关键就一点:根据客户端的情况返回最适合客户端的图片

那么,可能会存在哪些情况?在准备部署响应式图片的时候,我们可以问自己如下四个问题。

  • 是否希望根据客户端情况返回不同的图片 内容?
  • 是否希望根据客户端情况返回不同的图片 格式
  • 是否希望根据客户端情况返回不同的图片 尺寸
  • 是否希望优化高 分辨率 设备的体验?

picture 标签出来之前,这些只能通过 JS 来实现,不仅代码丑陋而且能力也不全。但是现在,针对这些问题,我们有了一个完整的优雅的解决方案。

picture 标签

picture 是 HTML5 新引入的标签,基本用法如下。

<picture>
  <source srcset="a.jpg" />
  <source srcset="b.jpg" />
  <img src="c.jpg" />
</picture>

我们可以这样理解,picture 标签会从 source 中选择最合适的一个,然后将它的 URL 赋值给 img。对于不认识 picture 的旧浏览器,他们会忽略 picture,只渲染 img,一切都不会有问题。

注意:picture 标签最后一定要包含一个 img 标签,否则,什么都不会显示。

现在我们逐一来看 picture 怎样解决上面的四个问题。

动态内容

根据客户端的情况,我们来返回完全不同的两张图。这个很简单,使用 source 标签的 media 属性即可。

如下代码会在小于 1024px 的时候显示 img-center.jpg,而在大于等于 1024px 的时候显示 img-full.jpg

<picture>
  <source
    media="(min-width: 1024px)"
    srcset="img-full.jpg"
  />

  <img src="img-center.jpg" />
</picture>

动态格式

这个问题也很简单,使用 source 标签的 type 属性即可。

如下代码会在支持 WebP 的浏览器上使用 img.webp,在不支持 WebP 的浏览器上使用 img.jpg

<picture>
  <source
    srcset="img.webp"
    type="image/webp"
  />

  <img src="img.jpg" />
</picture>

动态尺寸

如果希望浏览器能根据情况去请求不同尺寸的图片,我们需要提供两个信息:

  • 有哪些尺寸的图片
  • 图片显示的时候是什么尺寸

下面的代码中,我们首先使用 srcset 属性指定有哪些图片,分别是图片名和图片的尺寸,这里注意单位不用 px 而是 w,用于表示图片的固有宽度。

sizes 属性告诉浏览器,这个图片在不同的条件下会是什么样的宽度。这个属性用于给到浏览器提示,并不会真正的指定 img 的宽度,我们还是需要另外使用 CSS 来指定。

这样,给定一个视口宽度,浏览器可以得知图片需要的宽度,然后根据 DPI 情况,在所有可选图片中选择最合适的一个。

<img
  src="img-400.jpg"
  sizes="(min-width: 640px) 60vw, 100vw"
  srcset="img-200.jpg 200w, img-400.jpg 400w, img-800.jpg 800w, img-1200.jpg 1200w"
/>

动态分辨率

动态分辨率其实是动态尺寸的一种简化情况。

根据显示器的 DPI 返回同一张图片的不同分辨率版本可以直接利用 img 标签的 srcset 属性。

使用了如下的代码,浏览器会自动根据显示器的 DPI 来决定下载图片的哪个版本。

在低 DPI 设备上,例如桌面显示器,浏览器会使用 img-200.jpg,而在高 DPI 设备上,例如手机,浏览器会使用 img-400.jpg。

<img
  srcset="img-200.jpg, img-300.jpg 1.5x, img-400.jpg 2x"
  src="img-400.jpg"
/>

<style type="text/css">
  img {
    width: 200px;
  }
</style>

当然,我们也可以组合这几个选项。

如下的代码会

  • 视口 >= 1280px 时
    • 根据视口的具体宽度,返回不同尺寸的 img-full 图片
    • 如果客户端支持 WebP,返回 WebP 格式
  • 视口 < 1280px 时
    • 根据视口的具体宽度,返回不同尺寸的 img 图片
    • 如果客户端支持 WebP,返回 WebP 格式
<picture>
  <source
    media="(min-width: 1280px)"
    sizes="50vw"
    srcset="
      img-full-200.webp   200w,
      img-full-400.webp   400w,
      img-full-800.webp   800w,
      img-full-1200.webp 1200w,
      img-full-1600.webp 1600w,
      img-full-2000.webp 2000w
    "
    type="image/webp"
  />
  <source
    media="(min-width: 1280px)"
    sizes="50vw"
    srcset="img-full-200.jpg 200w, img-full-400.jpg 400w, img-full-800.jpg 800w, img-full-1200.jpg 1200w, img-full-1600.jpg 1800w, img-full-2000.jpg 2000w"
  />

  <source
    sizes="(min-width: 640px) 60vw, 100vw"
    srcset="img-200.webp 200w, img-400.webp 400w, img-800.webp 800w, img-1200.webp 1200w, img-1600.webp 1600w, img-2000.webp 2000w"
    type="image/webp"
  />
  <img
    src="img-400.jpg"
    sizes="(min-width: 640px) 60vw, 100vw"
    srcset="img-200.jpg 200w, img-400.jpg 400w, img-800.jpg 800w, img-1200.jpg 1200w, img-1600.jpg 1600w, img-2000.jpg 2000w"
  />
</picture>

这里强烈建议自己动手,结合 placeholder.com 网站,生成一些图片来测试,毕竟,纸上得来终觉浅。

参考