前端知识之性能优化——实现图片懒加载

懒加载,意思就是,在当前网页,滑动页面到能看到图片的时候再加载图片

故问题拆分成两个:

  1. 如何判断图片出现在了当前视口 (即如何判断我们能够看到图片)
  2. 如何控制图片的加载

方案一: 位置计算 + 滚动事件 (Scroll) + DataSet API

如何判断图片出现在了当前视口

  • offsetTop: 返回一个当前元素到其顶部的内边距的距离,只读属性 offsetTop = 自身margin + margin(多个父元素)+ padding(多个父元素) + border(多个父元素)
  • scrollTop:当前元素顶部到浏览器顶部的距离,即元素滚动出去的距离,如果该元素无垂直方向的滚动条,那么该元素的scrollTop为0,该元素可能是个小数

offsetTop - scrollTop <= 视口高度

js 复制代码
function isInViewPortOfOne (element) {
  // 获取可视窗口的高度。兼容所有浏览器
  const screenHeight = window.innerHeight || document.documentElement.clientHeight
  	 || document.body.clientHeight;
  // 获取滚动条滚动的高度
  const scrollTop = document.documentElement.scrollTop;
  // 获取元素偏移的高度。就是距离可视窗口的偏移量。
  const offsetTop = element.offsetTop;
  // 加100是为了提前加载
  return offsetTop - scrollTop <= screenHeight + 100;
}

如何控制图片的加载

<img data-src="shanyue.jpg" />

首先设置一个临时 Data 属性 data-src,控制加载时使用 src 代替 data-src,可利用 DataSet API 实现

img.src = img.datset.src

方案二: getBoundingClientRect API + Scroll with Throttle + DataSet API

如何判断图片出现在了当前视口

引入一个新的 API, Element.getBoundingClientRect() 方法返回元素的大小及其相对于视口的位置。

getBoundingClientRect()用于获得页面中某个元素的左,上,右和下分别相对浏览器视窗的位置。

getBoundingClientRect()是DOM元素到浏览器可视范围的距离(不包含文档卷起的部分)。

js 复制代码
// clientHeight 代表当前视口的高度
img.getBoundingClientRect().top < document.documentElement.clientHeight;

方案三: ## IntersectionObserver API + DataSet API

方案二使用的方法是: window.scroll 监听 Element.getBoundingClientRect() 并使用 _.throttle 节流

一系列组合动作太复杂了,于是浏览器出了一个三合一事件: IntersectionObserver API,一个能够监听元素是否到了当前视口的事件,一步到位!

事件回调的参数是 IntersectionObserverEntry(opens in a new tab) 的集合,代表关于是否在可见视口的一系列值

其中,entry.isIntersecting 代表目标元素可见

js 复制代码
let images = document.getElementsByTagName('img') //HTMLCollection
let father = document.getElementsByClassName('box')[0] //拥有滚动属性的dom
let fatherHeight = father.clientHeight 
let imgArr = [...images]
const callback = (entries) => {
  // console.log(entries)
  entries.forEach(entry => {
    // isIntersecting表示是否在交叉区域内(可见区域)
    if (entry.isIntersecting) {
        const image = entry.target;
        const data_src = image.getAttribute('data-src');
        image.setAttribute('src', data_src);
        observer.unobserve(image)
        console.log('触发')
     }
  })
}

const observer = new IntersectionObserver(callback); //创建观察对象
imgArr.forEach(image => {
   observer.observe(image); //开始观察
})

总结

第二、三种方法是浏览器API,存在兼容性问题。比较推荐使用第一种位置计算的方法,不存在兼容问题。

方案demo

html 复制代码
  <div class="box">
    <div class="blue"></div>
    <div class="red"></div>
    <div class="blue"></div>
    <div class="content">
        <img data-src="./1.jpg" height="300px" alt="" id="img1">
    </div>
  </div>
js 复制代码
let images = document.getElementsByTagName('img') //HTMLCollection
let father = document.getElementsByClassName('box')[0] //拥有滚动属性的dom
let fatherHeight = father.clientHeight 
let imgArr = [...images]
function lazyLoad() {
    imgArr.forEach(item => {
        // 方案一
        if (item.offsetTop + item.clientTop - father.scrollTop <= fatherHeight) {
            item.src = item.getAttribute('data-src')
         };
         // 方案二
        if(img.getBoundingClientRect().top < document.documentElement.clientHeight) {
            item.src = item.getAttribute('data-src')
         }
     })
}
   father.addEventListener('scroll', _.throttle(lazyLoad, 50))
相关推荐
崔庆才丨静觅6 小时前
hCaptcha 验证码图像识别 API 对接教程
前端
passerby60616 小时前
完成前端时间处理的另一块版图
前端·github·web components
掘了6 小时前
「2025 年终总结」在所有失去的人中,我最怀念我自己
前端·后端·年终总结
崔庆才丨静觅7 小时前
实用免费的 Short URL 短链接 API 对接说明
前端
崔庆才丨静觅7 小时前
5分钟快速搭建 AI 平台并用它赚钱!
前端
崔庆才丨静觅7 小时前
比官方便宜一半以上!Midjourney API 申请及使用
前端
Moment7 小时前
富文本编辑器在 AI 时代为什么这么受欢迎
前端·javascript·后端
崔庆才丨静觅8 小时前
刷屏全网的“nano-banana”API接入指南!0.1元/张量产高清创意图,开发者必藏
前端
剪刀石头布啊8 小时前
jwt介绍
前端
爱敲代码的小鱼8 小时前
AJAX(异步交互的技术来实现从服务端中获取数据):
前端·javascript·ajax