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 相关文章推荐
简单的JS多重继承示例
Mar 13 Javascript
JavaScript 事件冒泡简介及应用
Jan 11 Javascript
document.getElementById方法在Firefox与IE中的区别
May 18 Javascript
jquery简单实现鼠标经过导航条改变背景图
Dec 17 Javascript
JS实现鼠标滑过折叠与展开菜单效果代码
Sep 06 Javascript
去除字符串左右两边的空格(实现代码)
May 12 Javascript
JS实现选定指定HTML元素对象中指定文本内容功能示例
Feb 13 Javascript
从零开始学习Node.js系列教程四:多页面实现的数学运算示例
Apr 13 Javascript
BootStrap的两种模态框方式
May 10 Javascript
微信小程序开发之animation循环动画实现的让云朵飘效果
Jul 14 Javascript
JavaScript对象原型链原理详解
Feb 05 Javascript
解决Vue watch里调用方法的坑
Nov 07 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
浅析HTTP消息头网页缓存控制以及header常用指令介绍
2013/06/28 PHP
PHP使用token防止表单重复提交的方法
2016/04/07 PHP
laravel框架查询数据集转为数组的两种方法
2019/10/10 PHP
jquery 得到当前页面高度和宽度的两个函数
2010/02/21 Javascript
jquery如何判断某元素是否具备指定的样式
2013/11/05 Javascript
JS实现向表格中动态添加行的方法
2015/03/30 Javascript
jQuery仿淘宝网产品品牌隐藏与显示效果
2015/09/01 Javascript
JS+CSS实现仿支付宝菜单选中效果代码
2015/09/25 Javascript
实现非常简单的js双向数据绑定
2015/11/06 Javascript
vue.js入门(3)——详解组件通信
2016/12/02 Javascript
js中创建对象的几种方式
2017/02/05 Javascript
jQuery插件开发发送短信倒计时功能代码
2017/05/09 jQuery
Angularjs的启动过程分析
2017/07/18 Javascript
ES6中Array.includes()函数的用法
2017/09/20 Javascript
VSCode 配置React Native开发环境的方法
2017/12/27 Javascript
详解JavaScript中精度失准问题及解决方法
2020/02/04 Javascript
[56:41]2018DOTA2亚洲邀请赛 3.31 小组赛 A组 Newbee vs OG
2018/04/01 DOTA
django用户登录验证的完整示例代码
2019/07/21 Python
Pandas时间序列:重采样及频率转换方式
2019/12/26 Python
Python实现大数据收集至excel的思路详解
2020/01/03 Python
python调用win32接口进行截图的示例
2020/11/11 Python
html5与css3小应用
2013/04/03 HTML / CSS
美国著名的团购网站:Woot
2016/08/02 全球购物
国外最大的眼镜网站:Coastal
2017/08/09 全球购物
荷兰网上药店:Drogisterij.net
2019/09/03 全球购物
计算 s=(x*y)1/2,用两个宏定义来实现
2016/08/11 面试题
销售自荐信
2013/10/22 职场文书
口腔医学技术应届生求职信
2013/11/09 职场文书
学生拾金不昧表扬信
2014/01/21 职场文书
《恐龙》教学反思
2014/04/27 职场文书
妇联主席先进事迹
2014/05/18 职场文书
党员群众路线对照检查材料
2014/08/31 职场文书
中学生思想品德评语
2014/12/31 职场文书
倡议书格式及范文
2015/04/29 职场文书
2015年学校心理健康教育工作总结
2015/05/11 职场文书
阿里云日志过滤器配置日志服务
2022/04/09 Servers