页面加载完毕_手把手教你实现图片懒加载

a03a9e0ccd492b421c06effd64e59ebc.png

来源 | https://segmentfault.com/a/1190000024547276

懒加载(Lazy-Load)。它是针对图片加载时机的优化:在一些图片量比较大的网站(比如电商网站首页,或者团购网站、小游戏首页等),如果我们尝试在用户打开页面的时候,就把所有的图片资源加载完毕,那么很可能会造成白屏、卡顿等现象,因为图片真的太多了,一口气处理这么多任务,浏览器做不到啊! 目的 懒加载 的目的是当页面的图片进入到用户的可视范围之内在加载图片的一种优化方式。 可以增加首屏加载的速度,毕竟,用户点开页面的瞬间,呈现给他的只是首屏,我们只要把首屏的资源图片加载处理就可以了,至于下面的图片,当用户下滑当当前位置的时候,在加载出来也是没问题的,对于性能压力也小了,用户体验也没有变差。 原理 在页面初始化的时候,图片的src实际上是放在src属性上的,当元素处于可视范围内的时候,就把src赋值给src属性,完成图片加载。
// 在一开始加载的时候// 在进入可视范围内时

使用背景图来实现,原理也是一样的,把background-image放在,在可视范围时,就把src赋值给src属性,完成图片加载。
// 在一开始加载的时候// 在进入可视范围内时

实现一个懒加载

基于上面的实现思路,自己实现一个懒加载。

新建一个 index.html 中,为这些图片预置 img 标签:

      .img {      width: 200px;      height: 200px;      background-color: gray;      margin-bottom: 20px;    }    .pic {      width: 100%;      height: 100%;    }  
2a162da3c2d773dbbfc2e8b5b122132c.png
fc43543db873107192b734b63ea4a9e0.png
c455c450c86e02b2ade8d26f0340b7f5.png
0a882cc9eb5f6d30562f4e90e5a14dd8.png
0bcb33118dc335033fcaead960445ac7.png
40d9322fb9efbb08629494f72a6e4081.png
cb87de8490ef83919041644b32432920.png
55132bfa9babf13905113b0e801679bc.png
1ce3721902186ea9bc4b3cf8ab54bc0d.png
0ffe4ff0161f9d14f299f60d908a4ab5.png

在懒加载的实现中,有两个关键的数值:一个是当前可视区域的高度,另一个是元素距离可视区域顶部的高度。

当前可视区域的高度,在现代浏览器及 IE9 以上的浏览器中,可以使用window.innerHeight属性获取,在低版本的 IE 中使用document.documentElment.clientHeight 获取,这里我们兼容两种情况:

const viewHeight = window.innerHeight || document.documentElement.clientHeight;

而元素距离可视区域顶部的高度,这里我们用 getBoundingClientRect()方法来获取返回元素的大小和相对于尺寸的位置,对于该 API,MDN 的解释是:

Element.getBoundingClientRect() 方法返回元素的大小及其相对于视口的位置。

返回的属性中有一个相对于可视区域顶部的高度也就是top属性,刚好就是我们需要的元素距离顶部的距离。

这样,两个属性就都得到了。

我们利用当前可视区域的高度大于等于元素距离可视区域顶部的高度就可以确定,该元素是否已经进入到了可视范围:

9de34b12c5fe0c27d03fd2ca3cfe216b.png

  // 获取所有的图片标签  const imgs = document.getElementsByTagName("img");  // 获取可视区域的高度  const viewHeight =    window.innerHeight || document.documentElement.clientHeight;  // num用于统计当前显示到了哪一张图片,避免每次都从第一张图片开始检查是否露出  let num = 0;  function lazyload() {    console.log("滚动...");    for (let i = num; i < imgs.length; i++) {      // 用可视区域高度减去元素顶部距离可视区域顶部的高度      let distance = viewHeight - imgs[i].getBoundingClientRect().top;      // 如果可视区域高度大于等于元素顶部距离可视区域顶部的高度,说明元素露出      if (distance >= 0) {        // 给元素写入真实的src,展示图片        imgs[i].src = imgs[i].getAttribute("src");        // 前i张图片已经加载完毕,下次从第i+1张开始检查是否露出        num = i + 1;      }    }  }  // 防抖函数  function debounce(fn, delay = 500) {    let timer = null;    return function (...args) {      if (timer) clearTimeout(timer);      timer = setTimeout(() => {        fn.call(this, args);      }, delay);    };  }  // 是的页面初始化是加载首屏图片  window.onload = lazyload;  // 监听Scroll事件,为了防止频繁调用,使用防抖函数优化一下  window.addEventListener("scroll", debounce(lazyload, 600), false);

小结

  1. 先收集到页面中所有的img(也可以用class)。

  2. 获取到视图高度,计算显示的img,避免重复赋值src。

  3. 当滑动向下滑动鼠标,会触发onScroll事件(防抖),然后触发计算是否赋值src。

本文完~ e69dd5b386baf6ec0ccad11c52821340.png a9a23b9e29ff1027ad2171f6834708c4.png
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值