图片的javascript延时加载

在页面很长(超过3屏)且图片又很多时,默认情况下浏览器会加载所有图片,有可能导致第二屏的图片显示出来了,但第一屏的图片还在加载,这种情况最适合用"javascript延时加载"来改善用户体验.

原理:

1.根据图片元素距页面顶部的距离,判断图片自身在第几屏

2.所有图片元素的src值先不设置,改而用其它自定义属性,比如src写成lazy_src(这样浏览器就不会主动加载图片)

3.根据1的判断,如果轮到自己登场了(即到顶部的距离变化了,比如用户拉动滚动条了),则将lazy_src值赋值为src,这样浏览器就开始加载了.

代码:(收集于网上)

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
    <title>js延时加载</title>
    <script type="text/javascript">
        lazyLoad = (function () {
            var map_element = {};
            var element_obj = [];
            var download_count = 0;
            var last_offset = -1;
            var doc_body;
            var doc_element;
            var lazy_load_tag;
            
            function initVar(tags) {
                doc_body = document.body;
                //判断是否为IE的"怪异模式"
                doc_element = document.compatMode == 'BackCompat' ? doc_body : document.documentElement;
                lazy_load_tag = tags || ["img", "iframe"];
            };

            function initElementMap() {
                //var all_element = [];
                //从所有相关元素中找出需要延时加载的元素  
                for (var i = 0, len = lazy_load_tag.length; i < len; i++) {
                    var el = document.getElementsByTagName(lazy_load_tag[i]);
                    for (var j = 0, len2 = el.length; j < len2; j++) {
                        if (typeof (el[j]) == "object" && el[j].getAttribute("lazy_src")) {
                            element_obj.push(el[j]);
                        }
                    }
                }
				

                for (var i = 0, len = element_obj.length; i < len; i++) {
                    var o_img = element_obj[i];
                    var t_index = getAbsoluteTop(o_img); //得到图片相对document的距上距离  
                    if (map_element[t_index]) {
                        map_element[t_index].push(i);
                    } else {
                        //按距上距离保存一个队列  
                        var t_array = [];
                        t_array[0] = i;
                        map_element[t_index] = t_array;
                        download_count++; //需要延时加载的图片数量  
                    }
                }

            };

            function initDownloadListen() {
                if (!download_count) return;
                var offset = (window.MessageEvent && !document.getBoxObjectFor) ? doc_body.scrollTop : doc_element.scrollTop;
                //可视化区域的高=offtset+document的高  
                var visio_offset = offset + doc_element.clientHeight;
                if (last_offset == visio_offset) {
                    setTimeout(initDownloadListen, 200);//每隔0.2秒检测一次,这玩意儿感觉有些耗资源的
                    return;
                }
                last_offset = visio_offset;
                var visio_height = doc_element.clientHeight;
                var img_show_height = visio_height + offset;
                for (var key in map_element) {
                    if (img_show_height > key) {
                        var t_o = map_element[key];
                        var img_vl = t_o.length;
                        for (var l = 0; l < img_vl; l++) {
                            element_obj[t_o[l]].src = element_obj[t_o[l]].getAttribute("lazy_src");
                        }
                        delete map_element[key];
                        download_count--;
                    }
                }
                setTimeout(initDownloadListen, 200);
            };

            function getAbsoluteTop(element) {
                if (arguments.length != 1 || element == null) {
                    return null;
                }
                var offsetTop = element.offsetTop;
                while (element = element.offsetParent) {
                    offsetTop += element.offsetTop;
                }
                return offsetTop;
            }

            function init(tags) {
                initVar(tags);
                initElementMap();
                initDownloadListen();
            };

            return {
                init: init
            }
        })();
    </script>
	<style type="text/css">
		img{height:800px}
	</style>
</head>
<body>
    <div>
    <img lazy_src="http://pic.vlike.com/girl/UploadFiles_9401/201005/13/2010051307474029890_big.jpg" /><br/>
    <img lazy_src="http://pic.vlike.com/girl/UploadFiles_9401/201005/13/2010051307474372951_big.jpg" /><br/>
    <img lazy_src="http://pic.vlike.com/girl/UploadFiles_9401/201005/06/2010050616595471474_big.jpg" /><br/>
    <img lazy_src="http://pic.vlike.com/girl/UploadFiles_9401/201005/13/2010051307474693772_big.jpg" /><br/>
    <img lazy_src="http://pic.vlike.com/girl/UploadFiles_9401/201005/05/2010050511370456961_big.jpg" /><br/>
    <img lazy_src="http://pic.vlike.com/girl/UploadFiles_9401/201005/13/2010051311000442310_big.jpg" /><br/>
    </div>
    <script type="text/javascript">
        lazyLoad.init(); 
    </script>
</body>
</html>

注意事项:图片最好都用样式定义高度,否则图片的高度默认为0,就上面的例子来讲,所有img标签都挤在一起,全部在第一屏,导致程序认为它们都应该加载,就看不到效果了. 

另外这种方式有一个致命的缺点:如果浏览器禁用了javascript,将会失效!所以使用前请先考虑清楚,或者在页面上加一些提示(类似:“您的浏览器不支持javascript,页面显示可能不正常”之类)

附:firefox上快速禁用和启用javascript的方法

如上图所示,在firefox地址栏里输入about:config,找到javascript:enabled,双击可切换true或false(即:启用或禁用)

当然,园子里的高手还有一篇更强大的ImagesLazyLoad 图片延迟加载效果 推荐给想深入研究的朋友们看看。

此外,jQuery也一个专门实现延时加载的插件jQuery.lazyload,也很方便。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏前端侠2.0

表格内,设置许多元素的大小时,js的速度慢的办法

思路1:设置td为relative,input为absolute,然后设置input大小为td的大小。

15720
来自专栏计算机编程

SNS项目笔记<八>--Slides显示大图

在浏览大图的时候,我们总有些长图进行查看,在查看的过程中,我们难以一次性查看全部内容,于是我们想,如果在slides里面技能横滑切换,又能竖滑滚动查看就好了。这...

12120
来自专栏上善若水

010HTML页面加载和解析流程详细介绍

25150
来自专栏jiajia_deng

react-router 环境使用锚点的方法

28820
来自专栏张戈的专栏

解决WordPress4.4.1不支持图片暗箱问题,Begin主题适用

前几天更新 WordPress 最新 4.41 版本之后,发现文章之前的图片暗箱弹出功能没了,而且之前设置好是缩略图尺寸也没了,直接就是最大尺寸。 看了下当前页...

34540
来自专栏ionic3+

【Appetite】ionic3实录(二)UI分析及总体配置

首先,如果你是初学者,我强力建议你先看完我这篇文章: 【开发指南】(四)Ionic3快速上手并了解这些

13130
来自专栏jiajia_deng

react-router 环境使用锚点的方法

15740
来自专栏前端人人

React第三方组件1(路由管理之Router的使用②多层级跳转及重定向)

本教程总共6篇,每日更新一篇,请关注我们!你可以进入历史消息查看以往文章,也敬请期待我们的新文章! 1、React第三方组件1(路由管理之Router的使用①...

34740
来自专栏谦谦君子修罗刀

RN项目第一节

一、项目说明 本项目为模仿美团的项目,采用的是网上提供的API接口。导航采用 公司推荐的react-navigation,滚动条采用第三方组件react-na...

36660
来自专栏十月梦想

VUe入门详解之条件渲染

v-if和v-show效果一样,当不满足条件时候,页面不显示响应内容,但是v-if不满足条件,不挂载在dom中,而v-show只是dispaly:none隐藏了...

9620

扫码关注云+社区

领取腾讯云代金券