Javascript实现图片懒加载插件的方法


Posted in Javascript onOctober 20, 2016

前言

网络上各大论坛,尤其是一些图片类型的网站上,在图片加载时均采用了一种名为懒加载的方式,具体表现为,当页面被请求时,只加载可视区域的图片,其它部分的图片则不加载,只有这些图片出现在可视区域时才会动态加载这些图片,从而节约了网络带宽和提高了初次加载的速度,具体实现的技术并不复杂,下面分别对其说明。

Web 图片的懒加载就是通过读取img元素,然后获得img元素的data-src(也可以约定为其他属性名)属性的值,并赋予img的src,从而实现动态加载图片的机制。

这里需要注意的是: img在初始化的时候不要设置src属性,因为即使设置 src='' 浏览器也会尝试加载图片。

一个简单的图片懒加载共涉及两个方面

1. HTML 约定

我们首先需要给准备实施懒加载的img元素添加指定的class 这里为m-lazyload ,同时将img src赋值给 data-src属性。

具体示例为:

<img class="m-lazyload" data-src="imgUrl">

2. JavaScript 实现

动态加载总共分为以下几个步骤,这里每个步骤都将被拆分为独立的函数

1. 添加页面滚动监听事件

window.addEventListener('scroll', _delay, false);

function _delay() {
 clearTimeout(delay);
 delay = setTimeout(function () {
 _loadImage();
 }, time);
}

2. 当触发监听事件时会执行 _loadImage 函数,该函数负责加载图片

function _loadImage() {
 for (var i = imgList.length; i--;) {
 var el = imgList[i];
 if (_isShow(el)) {
  el.src = el.getAttribute('data-src');
  el.className = el.className.replace(new RegExp("(\\s|^)" + _selector.substring(1, _selector.length) + "(\\s|$)"), " ");
  imgList.splice(i, 1);
 }
 }
}

虽然执行了_loadImage函数,但是我们得知道哪些图片需要被加载,这里的判断依据是什么呢?

依据就是判断该图片是否在当前窗口的可视区域内,在这里我们封装一个_isShow函数来实现

function _isShow(el) {
 var coords = el.getBoundingClientRect();
 return ( (coords.top >= 0 && coords.left >= 0 && coords.top) <= (window.innerHeight || document.documentElement.clientHeight) + parseInt(offset));
}

自此一个图片加载的闭环就形成了

当网页滚动的事件被触发 -> 执行加载图片操作 -> 判断图片是否在可视区域内 -> 在,则动态将data-src的值赋予该图片。

太简单了?

事实就是如此!!!

如此简单,不妨扩展一下

添加一些自定义参数,谁都喜欢自定义,不是吗?

支持iScroll, iScroll是一个高性能,资源占用少,无依赖,多平台的javascript滚动插件。

这里我们做了些优化

图片加载后移除选择器,避免重复判断。

缓存img元素结合,减少dom元素查询性能损耗

扩展prototype添加getNode方法,支持分页数据懒加载(由于我们之前缓存了dom元素)

OK!说了这么多,show me the code 吧!

(function () {
 var imgList = [], // 页面所有img元素集合
 delay, // setTimeout 对象
 offset, //偏移量,用于指定图片距离可视区域多少距离,进行加载
 time, // 延迟载入时间
 _selector; // 选择器 默认为 .m-lazyload

 function _isShow(el) {
 var coords = el.getBoundingClientRect();
 return ( (coords.top >= 0 && coords.left >= 0 && coords.top) <= (window.innerHeight || document.documentElement.clientHeight) + parseInt(offset));
 }

 function _loadImage() {
 for (var i = imgList.length; i--;) {
  var el = imgList[i];
  if (_isShow(el)) {
  el.src = el.getAttribute('data-src');
  el.className = el.className.replace(new RegExp("(\\s|^)" + _selector.substring(1, _selector.length) + "(\\s|$)"), " ");
  imgList.splice(i, 1);
  }
 }
 }

 function _delay() {
 clearTimeout(delay);
 delay = setTimeout(function () {
  _loadImage();
 }, time);
 }

 function ImageLazyload(selector, options) {
 var defaults = options || {};
 offset = defaults.offset || 0;
 time = defaults.time || 250;
 _selector = selector || '.m-lazyload';
 this.getNode();
 _delay();//避免首次加载未触发touch事件,主动触发一次加载函数
 if (defaults.iScroll) {
  defaults.iScroll.on('scroll', _delay);
  defaults.iScroll.on('scrollEnd', _delay);
 } else {
  window.addEventListener('scroll', _delay, false);
 }
 }
 ImageLazyload.prototype.getNode = function () {
 imgList = [];
 var nodes = document.querySelectorAll(_selector);
 for (var i = 0, l = nodes.length; i < l; i++) {
  imgList.push(nodes[i]);
 }
 };
})();

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家学习或者使用图片懒加载的时候能有所帮助,如果有有疑问大家可以留言交流。

Javascript 相关文章推荐
微信小程序 实战小程序实例
Oct 08 Javascript
js图片切换具体实现代码
Oct 13 Javascript
JS基于面向对象实现的拖拽功能示例
Dec 20 Javascript
Javascript基础回顾之(二) js作用域
Jan 31 Javascript
js 作用域和变量详解
Feb 16 Javascript
详解react、redux、react-redux之间的关系
Apr 11 Javascript
Vue监听数据渲染DOM完以后执行某个函数详解
Sep 11 Javascript
Angular如何由模板生成DOM树的方法
Dec 23 Javascript
vue实现购物车的监听
Apr 20 Javascript
基于leaflet.js实现修改地图主题样式的流程分析
May 15 Javascript
Vue如何实现监听组件原生事件
Jul 03 Javascript
react项目从新建到部署的实现示例
Feb 19 Javascript
Vue.js Ajax动态参数与列表显示实现方法
Oct 20 #Javascript
探究Vue.js 2.0新增的虚拟DOM
Oct 20 #Javascript
Javascript 创建类并动态添加属性及方法的简单实现
Oct 20 #Javascript
javascript匀速动画和缓冲动画详解
Oct 20 #Javascript
js设置和获取自定义属性的方法
Oct 20 #Javascript
js阻止冒泡和默认事件(默认行为)详解
Oct 20 #Javascript
浅谈JS中String()与 .toString()的区别
Oct 20 #Javascript
You might like
php ss7.5的数据调用 (笔记)
2010/03/08 PHP
PHP调用C#开发的dll类库方法
2014/07/28 PHP
PHP实现对数字分隔加千分号的方法
2019/03/18 PHP
javascript 特殊字符串
2009/02/25 Javascript
javascript 使td内容不换行不撑开
2012/11/29 Javascript
在父页面调用子页面的JS方法
2013/09/29 Javascript
文本域中换行符的替换示例
2014/03/04 Javascript
详解JavaScript函数对象
2015/11/15 Javascript
基于jQuery实现弹出可关闭遮罩提示框实例代码
2016/07/18 Javascript
浅谈Vue.js 中的 v-on 事件指令的使用
2018/11/25 Javascript
vue2.0结合Element-ui实战案例
2019/03/06 Javascript
js最全的数组的降维5种办法(小结)
2020/04/28 Javascript
解决vue-router路由拦截造成死循环问题
2020/08/05 Javascript
JS实现无限轮播无倒退效果
2020/09/21 Javascript
Python实现拷贝多个文件到同一目录的方法
2016/09/19 Python
Python3.6 Schedule模块定时任务(实例讲解)
2017/11/09 Python
Python实现的求解最大公约数算法示例
2018/05/03 Python
Python 使用PIL numpy 实现拼接图片的示例
2018/05/08 Python
python抓取网站的图片并下载到本地的方法
2018/05/22 Python
Flask框架WTForm表单用法示例
2018/07/20 Python
pandas中的series数据类型详解
2019/07/06 Python
pytorch的梯度计算以及backward方法详解
2020/01/10 Python
keras tensorflow 实现在python下多进程运行
2020/02/06 Python
Python OrderedDict字典排序方法详解
2020/05/21 Python
详解python metaclass(元类)
2020/08/13 Python
HTML5中实现拖放效果无须借助javascript
2012/12/26 HTML / CSS
蔬菜基地的创业计划书
2014/01/06 职场文书
班风学风建设方案
2014/05/06 职场文书
我爱祖国演讲稿
2014/09/02 职场文书
十一国庆节“向国旗敬礼”主题班会活动方案
2014/09/27 职场文书
森马旗舰店双十一营销方案
2014/09/29 职场文书
2015初中教导处工作总结
2015/07/21 职场文书
斗罗大陆八大特殊魂兽,龙族始祖排榜首,第五最残忍(翠魔鸟)
2022/03/18 国漫
《杜鹃的婚约》OP主题曲「凸凹」无字幕影像公开
2022/04/08 日漫
讲解MySQL增删改操作
2022/05/06 MySQL
MySQL数据库 安全管理
2022/05/06 MySQL