JavaScript实现图片懒加载的方法分析


Posted in Javascript onJuly 05, 2018

本文实例讲述了JavaScript实现图片懒加载的方法。分享给大家供大家参考,具体如下:

懒加载是非常实用的提升网页性能的方式,当访问一个页面的时候,只显示可视区域内的图片,其它的图片只有出现在可视区域内的时候才会被请求加载。

我们现在用原生的js实现简单的图片懒加载,主要利用的原理就是先不给设置src,而是把图片的路径放在data-src中,等待图片被加载的时候将路径取出放到src中。

HTML代码

<div class="container">
 <div class="img-area">
  <img class="my-photo" alt="loading" data-src="img/img1.png">
 </div>
 <div class="img-area">
  <img class="my-photo" alt="loading" data-src="img/img2.png">
 </div>
 <div class="img-area">
  <img class="my-photo" alt="loading" data-src="img/img3.png">
 </div>
 <div class="img-area">
  <img class="my-photo" alt="loading" data-src="img/img4.png">
 </div>
 <div class="img-area">
  <img class="my-photo" alt="loading" data-src="img/img5.png">
 </div>
</div>

判断元素是否在可视区域

方法一:

1. 获取屏幕可视区高度:document.documentElement.clientHeight
2. 获取元素距顶部的高度:element.offsetTop
3. 获取滚动高度:document.documentElement.scrollTop
4. 若满足:2-3<1,那么元素就出现在可视区域

方法二:

1. 获取元素到可视区域顶部的距离:var bound = element.getBoundingClientRect()
2. 获取可视区域的高度:window.innerHeight
3. 若满足bound.top<=window.innerHeight,那么元素就出现在可视区域

方法三:

利用IntersectionObserver函数自动观察元素是否在可视区域内

var watch = new IntersectionObserver(callback,option);
//开始观察
watch.observe(el);
//停止观察
watch.unobserve(el);
//关闭观察器
watch.disconnect();

js代码

第一种很多人都用过,所以我们就用第二种写一下

//判断图片是否出现在可视区域内
function isInSight(el) {
    const bound = el.getBoundingClientRect();
    const clientHeight = window.innerHeight;
    return bound.top <= clientHeight + 100;
}
//加载图片
let index = 0;
function checkImgs() {
    const imgs = document.querySelectorAll('.my-photo');
    for( let i = index; i < imgs.length; i++){
      if(isInSight(imgs[i])){
        loadImg(imgs[i]);
        index = i;
      }
    }
}
function loadImg(el) {
    if(!el.src){
      const source = el.dataset.src;
      el.src = source;
    }
}
//函数节流
//函数节流是很重要的思想,可以防止过于频繁的操作dom
function throttle(fn,mustRun = 500) {
    const timer = null;
    let previous = null;
    return function () {
      const now = new Date();
      const context = this;
      const args = arguments;
      if(!previous){
        previous = now;
      }
      const remaining = now -previous;
      if(mustRun && remaining >= mustRun){
        fn.apply(context,args);
        previous = now;
      }
    }
  }
//调用函数
window.onload=checkImgs;
window.onscroll = throttle(checkImgs);

我们在用第三种方法写一个demo

function checkImgs() {
 const imgs = Array.from(document.querySelectorAll(".my-photo"));
 imgs.forEach(item => io.observe(item));
}
function loadImg(el) {
 if (!el.src) {
  const source = el.dataset.src;
  el.src = source;
 }
}
const io = new IntersectionObserver(ioes => {
 ioes.forEach(ioe => {
  const el = ioe.target;
  const intersectionRatio = ioe.intersectionRatio;
  if (intersectionRatio > 0 && intersectionRatio <= 1) {
   loadImg(el);
  }
  el.onload = el.onerror = () => io.unobserve(el);
 });
});

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
js综合应用实例简单的表格统计
Sep 03 Javascript
往光标所在位置插入值的js代码
Sep 22 Javascript
jQuery javaScript捕获回车事件(示例代码)
Nov 07 Javascript
jQuery标签替换函数replaceWith()的使用例子
Aug 28 Javascript
AngularJS学习笔记之ng-options指令
Jun 16 Javascript
jQuery在ie6下无法设置select选中的解决方法详解
Sep 20 Javascript
Bootstrap3下拉菜单的实现
Feb 22 Javascript
详解Vue中一种简易路由传参办法
Sep 15 Javascript
Java设计中的Builder模式的介绍
Mar 22 Javascript
Vue中使用ElementUI使用第三方图标库iconfont的示例
Oct 11 Javascript
微信小程序 wx:for遍历循环使用实例解析
Sep 09 Javascript
原生js+css实现tab切换功能
Sep 17 Javascript
JavaScript实现浅拷贝与深拷贝的方法分析
Jul 05 #Javascript
手把手教你用Node.js爬虫爬取网站数据的方法
Jul 05 #Javascript
vue使用ElementUI时导航栏默认展开功能的实现
Jul 04 #Javascript
vue两个组件间值的传递或修改方式
Jul 04 #Javascript
jQuery实现炫丽的3d旋转星空效果
Jul 04 #jQuery
jQuery实现table表格checkbox全选的方法分析
Jul 04 #jQuery
React组件内事件传参实现tab切换的示例代码
Jul 04 #Javascript
You might like
ajax返回值中有回车换行、空格的解决方法分享
2013/10/24 PHP
php中rename函数用法分析
2014/11/15 PHP
学习php设计模式 php实现观察者模式(Observer)
2015/12/09 PHP
PHP命令空间namespace及use的用法小结
2017/11/27 PHP
PHP+Apache实现二级域名之间共享cookie的方法
2019/07/24 PHP
Laravel 6 将新增为指定队列任务设置中间件的功能
2019/08/06 PHP
JS 容错处理代码, 屏蔽错误信息
2021/03/09 Javascript
js constructor的实际作用分析
2011/11/15 Javascript
JS的replace方法介绍
2012/10/20 Javascript
jquery 提示信息显示后自动消失的具体实现
2013/12/18 Javascript
jQuery实现鼠标经过提示信息的地图热点效果
2015/04/26 Javascript
JS+CSS相对定位实现的下拉菜单
2015/10/06 Javascript
javascript电商网站抢购倒计时效果实现
2015/11/19 Javascript
node网页分段渲染详解
2016/09/05 Javascript
JS实现简单抖动效果
2017/06/01 Javascript
JavaScript实现的斑马线表格效果【隔行变色】
2017/09/18 Javascript
基于jQuery中ajax的相关方法汇总(必看篇)
2017/11/08 jQuery
原生JS实现ajax与ajax的跨域请求实例
2017/12/01 Javascript
Vue进度条progressbar组件功能
2018/04/17 Javascript
vue使用原生js实现滚动页面跟踪导航高亮的示例代码
2018/10/25 Javascript
JS XMLHttpRequest原理与使用方法深入详解
2020/04/30 Javascript
解决vue单页面 回退页面 keeplive 缓存问题
2020/07/22 Javascript
vue中使用vue-pdf的方法详解
2020/09/05 Javascript
Python translator使用实例
2008/09/06 Python
python打印直角三角形与等腰三角形实例代码
2019/10/20 Python
Python assert关键字原理及实例解析
2019/12/13 Python
Python 时间戳之获取整点凌晨时间戳的操作方法
2020/01/28 Python
Python实现初始化不同的变量类型为空值
2020/06/02 Python
一款纯css3实现的动画加载导航
2014/10/08 HTML / CSS
浅析与CSS3的loading动画加载相关的transition优化
2015/05/18 HTML / CSS
浅谈关于html5中图片抛物线运动的一些心得
2018/01/09 HTML / CSS
法人授权委托书格式
2014/04/08 职场文书
历史学专业求职信
2014/06/19 职场文书
优秀大学生申请书
2019/06/24 职场文书
使用pandas模块实现数据的标准化操作
2021/05/14 Python
Java基础——Map集合
2022/04/01 Java/Android