利用原生JS實現(xiàn)懶加載lazyLoad的三種方法總結(jié)
前言
首先我們先搭建好頁面如下:

<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
>

我們知道,圖片懶加載是在滾動條向下滾動時,才判斷圖片是否到達可視區(qū)域
于是我們需要在滾動監(jiān)聽時判斷圖片是否即將顯示,所以我們需要將圖片的真實地址先隱藏起來,即采用自定義屬性 data-src 保存圖片的真實地址,當滾動條滾動到圖片能夠看到時再加載真實的地址.
下面我們來看第一個方法
Method 1: 高度對比
這里我們采用 (元素距頂部的高度 - 頁面被卷去的高度 <= 瀏覽器可視區(qū)的高度) 來判斷是否符合我們想要的條件.這里我們需要實時監(jiān)聽頁面滾動時 圖片的高度變化
/**
* 方法一
* 高度對比
*/
let imgs = [...document.querySelectorAll('img')]; // 先獲取所有的圖片
window.addEventListener('scroll', function () {
})
添加完事件后再繼續(xù)判斷 圖片是否達到要求,即
/**
* 方法一
* 高度對比
*/
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)為真實地址
}
}
}
看起來還挺簡單的對吧? 不過我們還有更簡單的方法,如下:
Method 2: 使用getBoundingClientRect() 的API
先附上MDN 對getBoundingClientRect() 的解釋getBoundingClientRect()

我們可以通過 getBoundingClientRect().top來獲取元素距視口頂部的距離,于是我們就可以比較getBoundingClientRect().top 和 window.innerHeight 的值的關(guān)系來實現(xiàn)懶加載的效果
這里使用了getAttribute() 和setAttribute() 屬性
/**
* 方法二
* @params getBoundingClientRect()
* 可視區(qū)API
*/
let imgs = [...document.querySelectorAll('img')];
window.addEventListener('scroll', function () {
imgs.forEach(img => {
//這里其實和Method1的思想一樣,不過就是簡潔了一些
if (img.getBoundingClientRect().top < window.innerHeight) {
let dataSrc = img.getAttribute(' data-src'); // 獲取 data-src 真實的地址
img.setAttribute('src', dataSrc); // 將 真實的地址 替換為 src屬性
console.log(img.src);
}
})
})
Method 3: 采用最新的 IntersectionObserver 構(gòu)造函數(shù)
看過上面兩種方法,那你是否覺得懶加載還挺簡單的對吧?
沒錯,我們寫的代碼很少,看起來很簡單,但是我們忽略了一個重要的問題:
圖片替換為真實的地址之后,如果我們反復(fù)的拉動滾動條,會一直觸發(fā) if()條件,
所以我在 Method2 方法里給了一個 console.log(img.src);
目的就是為了讓你看到當有人持續(xù)不斷的拉動滾動條,會一直打印 console.log(img.src);
那我們怎么去讓圖片真實地址加載完之后,不再觸發(fā)對它的頻繁操作呢?或者說怎么優(yōu)化游覽器的性能呢?
好巧不巧,現(xiàn)在有了一個新增的構(gòu)造函數(shù),來解決我們的頻繁觸發(fā)條件語句的問題.
這個構(gòu)造函數(shù)就是 IntersectionObserver
根據(jù) MDN 上的解釋
- IntersectionObserver()構(gòu)造器創(chuàng)建并返回一個IntersectionObserver對象。 如果指定rootMargin則會檢查其是否符合語法規(guī)定,檢查閾值以確保全部在0.0到1.0之間,并且閾值列表會按升序排列。如果閾值列表為空,則默認為一個[0.0]的數(shù)組。
- callback當元素可見比例超過指定閾值后,會調(diào)用一個回調(diào)函數(shù),此回調(diào)函數(shù)接受兩個參數(shù):
entries一個IntersectionObserverEntry對象的數(shù)組,每個被觸發(fā)的閾值,都或多或少與指定閾值有偏差。
observer被調(diào)用的IntersectionObserver實例。
這里我們只使用第一個參數(shù) callback 這個回調(diào)函數(shù)
window.addEventListener('scroll', function () {
// 首先我們先實例化這個構(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)它是個數(shù)組,于是
entries.forEach(ele => {
console.log(ele); // 我們再打印一下元素,看看元素里面有什么
// 如下圖
})
};
}

我們找到了 isIntersecting: false 這個屬性,這個意思是 是否交叉,根據(jù)構(gòu)造函數(shù)的意義我們得知,交叉可以理解為是否被觀察到
如果被觀察到, 那我們就讓他的真實地址替換為 它的 src 屬性 ,并且取消對它的觀察
/**
* 方法三
* @params new IntersectionObserver(callback[,options])
* 觀察-->構(gòu)造函數(shù)
*/
window.addEventListener('scroll', function () {
let imgs = [...document.querySelectorAll('.img')]
const callback = entries => { // entries 是觀察的元素數(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 是目標元素
observe.unobserve(ele.target) // 真實地址替換后 取消對它的觀察
}
})
};
const observe = new IntersectionObserver(callback); // 實例化 IntersectionObserver
imgs.forEach(image => {
observe.observe(image) // observe : 被調(diào)用的IntersectionObserver實例。給每個圖片添加觀察實例
})
}
這樣處理,我們就可以不再頻繁的去觸發(fā) if() 條件語句
因為在圖片替換了真實地址后,我取消了對當前圖片的觀察,于是,當前圖片已經(jīng)沒有事件再被觸發(fā),所以這樣對瀏覽器的性能進行了極大的優(yōu)化
總結(jié)
到此這篇關(guān)于利用原生JS實現(xiàn)懶加載lazyLoad的三種方法的文章就介紹到這了,更多相關(guān)JS實現(xiàn)懶加載lazyLoad內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用Plupload實現(xiàn)直接上傳附件至七牛云存儲
這篇文章主要介紹了使用Plupload實現(xiàn)直接上傳附件至七牛云存儲,需要的朋友可以參考下2014-12-12
JavaScript中的普通函數(shù)和箭頭函數(shù)的區(qū)別和用法詳解
這篇文章主要介紹了JavaScript中的普通函數(shù)和箭頭函數(shù)的區(qū)別和用法詳解,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-03-03
swiperjs實現(xiàn)導(dǎo)航與tab頁的聯(lián)動
這篇文章主要為大家詳細介紹了swiperjs實現(xiàn)導(dǎo)航與tab頁的聯(lián)動,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-12-12

