亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

利用原生JS實(shí)現(xiàn)懶加載lazyLoad的三種方法總結(jié)

 更新時(shí)間:2021年07月05日 10:57:11   作者:JayDo257248  
加載頁面的時(shí)候,圖片一直都是流量大頭,針對圖片的性能方法也挺多的比如base64、雪碧圖等;懶加載也是其中一種,這篇文章主要給大家介紹了關(guān)于利用原生JS實(shí)現(xiàn)懶加載lazyLoad的三種方法,需要的朋友可以參考下

前言

首先我們先搭建好頁面如下:

<style>
    * {
      padding: 0%;
      margin: 0%;
    }

    .contain img {
      width: 600px;
      height: 400px;
    }

    ul {
      list-style: none;
    }

    li {
      display: inline-block;
    }
  </style>
  
  
  <div class="contain">
    <ul>
       <li><img data-src="./images/iu.jpg"  src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu1.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu2.png" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu3.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu4.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu5.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu6.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu7.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu8.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu9.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/iu10.jpg" src='./images/lazy.png' alt=""></li>
      <li><img data-src="./images/zzf_01.jpg" src='./images/lazy.png' alt=""></li>
    </ul>
  </div
>

我們知道,圖片懶加載是在滾動(dòng)條向下滾動(dòng)時(shí),才判斷圖片是否到達(dá)可視區(qū)域

于是我們需要在滾動(dòng)監(jiān)聽時(shí)判斷圖片是否即將顯示,所以我們需要將圖片的真實(shí)地址先隱藏起來,即采用自定義屬性 data-src 保存圖片的真實(shí)地址,當(dāng)滾動(dòng)條滾動(dòng)到圖片能夠看到時(shí)再加載真實(shí)的地址.

下面我們來看第一個(gè)方法

Method 1: 高度對比

這里我們采用 (元素距頂部的高度 - 頁面被卷去的高度 <=  瀏覽器可視區(qū)的高度) 來判斷是否符合我們想要的條件.這里我們需要實(shí)時(shí)監(jiān)聽頁面滾動(dòng)時(shí) 圖片的高度變化

/**
 * 方法一
 * 高度對比
 */
 let imgs = [...document.querySelectorAll('img')]; // 先獲取所有的圖片
 window.addEventListener('scroll', function () {
   
 })

添加完事件后再繼續(xù)判斷 圖片是否達(dá)到要求,即

 /**
  * 方法一
  * 高度對比
  */
  let imgs = [...document.querySelectorAll('img')]; // 先獲取所有的圖片
  window.addEventListener('scroll', function () {
    lazyLoad(imgs)
  })
  function lazyLoad(imgs) {
    for (var i = 0; i < imgs.length; i++) {
      var height = imgs[i].offsetTop; // 圖片的距頂部的高度
      var wheight = window.innerHeight; // 瀏覽器可視區(qū)的高度
      var sheight = document.documentElement.scrollTop; // 頁面被卷去的高度
      if (height - sheight <= wheight) { // 判斷圖片是否將要出現(xiàn)
        imgs[i].src = imgs[i].dataset.src; // 出現(xiàn)后將自定義地址轉(zhuǎn)為真實(shí)地址
      }
    }
  }

看起來還挺簡單的對吧? 不過我們還有更簡單的方法,如下:

Method 2: 使用getBoundingClientRect() 的API

先附上MDN 對getBoundingClientRect() 的解釋getBoundingClientRect()

我們可以通過 getBoundingClientRect().top來獲取元素距視口頂部的距離,于是我們就可以比較getBoundingClientRect().top 和  window.innerHeight 的值的關(guān)系來實(shí)現(xiàn)懶加載的效果
這里使用了getAttribute()setAttribute() 屬性

 /**
  * 方法二
  * @params getBoundingClientRect()
  * 可視區(qū)API
  */

 let imgs = [...document.querySelectorAll('img')];
 window.addEventListener('scroll', function () {
    imgs.forEach(img => { 
      //這里其實(shí)和Method1的思想一樣,不過就是簡潔了一些
      if (img.getBoundingClientRect().top < window.innerHeight) { 
     
        let dataSrc = img.getAttribute(' data-src'); // 獲取 data-src 真實(shí)的地址
        img.setAttribute('src', dataSrc); // 將 真實(shí)的地址 替換為 src屬性
        console.log(img.src);
      }
    })
  })

Method 3: 采用最新的 IntersectionObserver 構(gòu)造函數(shù)

看過上面兩種方法,那你是否覺得懶加載還挺簡單的對吧?

沒錯(cuò),我們寫的代碼很少,看起來很簡單,但是我們忽略了一個(gè)重要的問題:

    圖片替換為真實(shí)的地址之后,如果我們反復(fù)的拉動(dòng)滾動(dòng)條,會(huì)一直觸發(fā) if()條件,
   
    所以我在 Method2 方法里給了一個(gè)   console.log(img.src);
   
    目的就是為了讓你看到當(dāng)有人持續(xù)不斷的拉動(dòng)滾動(dòng)條,會(huì)一直打印 console.log(img.src);

那我們怎么去讓圖片真實(shí)地址加載完之后,不再觸發(fā)對它的頻繁操作呢?或者說怎么優(yōu)化游覽器的性能呢?

好巧不巧,現(xiàn)在有了一個(gè)新增的構(gòu)造函數(shù),來解決我們的頻繁觸發(fā)條件語句的問題.

這個(gè)構(gòu)造函數(shù)就是 IntersectionObserver

根據(jù) MDN 上的解釋

  • IntersectionObserver()構(gòu)造器創(chuàng)建并返回一個(gè)IntersectionObserver對象。 如果指定rootMargin則會(huì)檢查其是否符合語法規(guī)定,檢查閾值以確保全部在0.0到1.0之間,并且閾值列表會(huì)按升序排列。如果閾值列表為空,則默認(rèn)為一個(gè)[0.0]的數(shù)組。
  • callback當(dāng)元素可見比例超過指定閾值后,會(huì)調(diào)用一個(gè)回調(diào)函數(shù),此回調(diào)函數(shù)接受兩個(gè)參數(shù):
    entries一個(gè)IntersectionObserverEntry對象的數(shù)組,每個(gè)被觸發(fā)的閾值,都或多或少與指定閾值有偏差。
    observer被調(diào)用的IntersectionObserver實(shí)例。

這里我們只使用第一個(gè)參數(shù) callback 這個(gè)回調(diào)函數(shù)

window.addEventListener('scroll', function () {
// 首先我們先實(shí)例化這個(gè)構(gòu)造函數(shù)
   const observe = new IntersectionObserver(callback);
// 然后寫我們需要處理業(yè)務(wù)的回調(diào)函數(shù) callback
   const callback = entries => {
        console.log(entries); //我們先打印一下 entries 看看有什么用 
        // 如下圖
    };
}

window.addEventListener('scroll', function () {
   const observe = new IntersectionObserver(callback);
// 然后寫我們需要處理業(yè)務(wù)的回調(diào)函數(shù) callback
   const callback = entries => {
       // 我們發(fā)現(xiàn)它是個(gè)數(shù)組,于是
       entries.forEach(ele => {
           console.log(ele); // 我們再打印一下元素,看看元素里面有什么
           // 如下圖
       })
    };
}

我們找到了 isIntersecting: false 這個(gè)屬性,這個(gè)意思是 是否交叉,根據(jù)構(gòu)造函數(shù)的意義我們得知,交叉可以理解為是否被觀察到

如果被觀察到, 那我們就讓他的真實(shí)地址替換為 它的 src 屬性 ,并且取消對它的觀察

 /**
  * 方法三
  * @params   new IntersectionObserver(callback[,options])
  * 觀察-->構(gòu)造函數(shù)
  */
  window.addEventListener('scroll', function () {
    let imgs = [...document.querySelectorAll('.img')]
    const callback = entries => { // entries 是觀察的元素?cái)?shù)組
        entries.forEach(ele => {
            if (ele.isIntersecting) { // isIntersecting  是否被觀察到
                const data_src = ele.target.getAttribute('data-src'); //這里基本和 Method1/Method2一樣
                ele.target.setAttribute('src', data_src); // ele.target 是目標(biāo)元素
                observe.unobserve(ele.target) // 真實(shí)地址替換后 取消對它的觀察
            }
        })
    };
    const observe = new IntersectionObserver(callback); // 實(shí)例化 IntersectionObserver
    imgs.forEach(image => { 
        observe.observe(image) // observe : 被調(diào)用的IntersectionObserver實(shí)例。給每個(gè)圖片添加觀察實(shí)例
    })
  }

這樣處理,我們就可以不再頻繁的去觸發(fā) if() 條件語句

因?yàn)樵趫D片替換了真實(shí)地址后,我取消了對當(dāng)前圖片的觀察,于是,當(dāng)前圖片已經(jīng)沒有事件再被觸發(fā),所以這樣對瀏覽器的性能進(jìn)行了極大的優(yōu)化

總結(jié)

到此這篇關(guān)于利用原生JS實(shí)現(xiàn)懶加載lazyLoad的三種方法的文章就介紹到這了,更多相關(guān)JS實(shí)現(xiàn)懶加載lazyLoad內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評論