zoukankan      html  css  js  c++  java
  • 图片懒加载

    图片懒加载

    为什么要懒加载

    原文出处:https://juejin.im/post/5aa68f226fb9a028dc40ac5d

    有时候页面中会有很多图片,一个图片就意味着一个http请求,一个js文件几十k,一张图片就可能是几M,过多的图片需要很长的加载时间,等图片加载完成,早就人走茶凉,图片懒加载对提升用户体验有着显著的效果

    懒加载的原理

    当图片不在可视区域内时,统一设置imgsrc为指定图片src='default.png',添加data-src(自定义属性)属性指向真实图片url 

    <img src='default.png' data-src='httt://www.xxx.com/images/001.jpg'>
    

    监听scroll事件,当图片出现在可视区时,提取data-src的值并赋给src,加载真正图片

    实现代码

            let img = document.getElementsByTagName('img')
            //设置每次遍历的起始图片,防止重复加载
            let n = 0
            //加载可视区域图片
            lazyLoad()
            window.onscroll = lazyLoad
            function lazyLoad() {
                let seeHeight = document.documentElement.clientHeight
                let scrollHeight = document.body.scrollHeight
                for (let i = n; i<img.length;i++) {
                   if(img[i].src = 'default.png'){
                       if(img[i].offsetHeight < seeHeight + scrollHeight){
                           img[i].setAttribute('src') = img[i].getAttribute('data-src')
                           n++
                       }
                   }
                }
            }
    

    还没完,lazyLodescorll事件绑定会导致高频触发,这只会增加浏览器的压力,违背了我们的初衷,所以我们需要限制事件频率,改良代码如下

           let img = document.getElementsByTagName('img')
            //设置每次遍历的起始图片,防止重复加载
            let n = 0
            //加载可视区域图片
            lazyLoad()
            function lazyLoad() {
                let seeHeight = document.documentElement.clientHeight
                let scrollHeight = document.body.scrollHeight
                for (let i = n; i<img.length;i++) {
                   if(img[i].src = 'default.png'){
                       if(img[i].offsetHeight < seeHeight + scrollHeight){
                           img[i].setAttribute('src') = img[i].getAttribute('data-src')
                           n++
                       }
                   }
                }
            }
            //防抖
            function debounce(fn,wait) {
                let timeout = null
                return function(){ 
                    let context = this
                    let args = arguments
                    clearTimeout(timeout)
                    timeout = setTimeout( function(){
                      fn.apply(context,args)
                    },wait)                    
                }
            }
            window.addEventListener('scroll',debounce(lazyLoad,800),true)
    

    debounce函数限制了lazyLoad的触发频率,800ms等待时间内scroll时间再次触发则重置时间,术语叫防抖。这就完了?nonono!假设我们把wait设的大点,2s,如果用户一直滑动滚动条,时间不断被重置,造成的效果是lazyLoad一直不被执行,图片加载不出来,这是不能接受的,所以我们需要设置一个时间,超过该时间lazyLoad必须执行一次,术语叫节流,代码如下

           let img = document.getElementsByTagName('img')
            //设置每次遍历的起始图片,防止重复加载
            let n = 0
            //加载可视区域图片
            lazyLoad()
            function lazyLoad() {
                let seeHeight = document.documentElement.clientHeight
                let scrollHeight = document.body.scrollHeight
                for (let i = n; i<img.length;i++) {
                   if(img[i].src = 'default.png'){
                       if(img[i].offsetHeight < seeHeight + scrollHeight){
                           img[i].setAttribute('src') = img[i].getAttribute('data-src')
                           n++
                       }
                   }
                }
            }
            //节流
            function throttle(fn,wait,mustTime){
                let timeout = null
                let startTime = new Date()
                let curTime
                return function(){
                    let context = this
                    let args = arguments
                    curTime = new Date()
                    if( (curTime - startTime) >= mustTime){
                         startTime = curTime
                        fn.apply(context,args) 
                        clearTimeout(timeout)    
                    }else{
                        clearTimeout(timeout)
                        timeout = setTimeout( function(){
                            fn.apply(context,args)
                            startTime = new Date()
                        },wait)
                    }
                }
            }
            window.addEventListener('scroll',throttle( lazyLoad,2000,3000))
    分享文章如有侵权,版权等问题,请私信联系我,我将第一时间删除或修正。
  • 相关阅读:
    centos 6.8 配置 Redis3.2.5
    php将字符串转为二进制数据串
    php密码对称encrypt加密
    mysql 新建用户并赋予远程访问权限
    centos6.8 搭建postfix/dovecot邮件服务器
    centos 6.8 设置svn钩子同步至web目录
    百分百解决Job for network.service failed. See 'system的问题
    java锁机制详解
    解决网页打不开简书的问题
    SpringBoot读取不到application.yml
  • 原文地址:https://www.cnblogs.com/mhtss/p/8569177.html
Copyright © 2011-2022 走看看