本文標題:實現圖片懶加載(lazyload)
文章做者:Jake
發佈時間:2016-11-26, 18:46:34
最後更新:2016-11-28, 17:12:59
原始連接:i.jakeyu.top/2016/11/26/…
許可協議: "署名-非商用-相同方式共享 4.0" 轉載請保留原文連接及做者。javascript
對頁面加載速度影響最大的就是圖片,一張普通的圖片能夠達到幾M的大小,而代碼也許就只有幾十KB。當頁面圖片不少時,頁面的加載速度緩慢,幾S鍾內頁面沒有加載完成,也許會失去不少的用戶。css
因此,對於圖片過多的頁面,爲了加速頁面加載速度,因此不少時候咱們須要將頁面內未出如今可視區域內的圖片先不作加載, 等到滾動到可視區域後再去加載。這樣子對於頁面加載性能上會有很大的提高,也提升了用戶體驗。html
將頁面中的img標籤src指向一張小圖片或者src爲空,而後定義data-src
(這個屬性能夠自定義命名,我才用data-src)屬性指向真實的圖片。src
指向一張默認的圖片,不然當src
爲空時也會向服務器發送一次請求。能夠指向loading
的地址。java
注:圖片要指定寬高jquery
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" />複製代碼
當載入頁面時,先把可視區域內的img標籤的data-src
屬性值負給src
,而後監聽滾動事件,把用戶即將看到的圖片加載。這樣便實現了懶加載。瀏覽器
在寫代碼前,須要瞭解各類高度。先看這篇文章scrollTop,offsetTop,scrollLeft,offsetLeft性能優化
<head>
<meta charset="UTF-8">
<title>Document</title>
<style> img { display: block; margin-bottom: 50px; width: 400px; height: 400px; } </style>
</head>
<body>
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww1.sinaimg.cn/large/006y8mN6gw1fa7kaed2hpj30sg0l9q54.jpg" alt="">
<img src="default.jpg" data-src="http://ww1.sinaimg.cn/large/006y8mN6gw1fa7kaed2hpj30sg0l9q54.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww1.sinaimg.cn/large/006y8mN6gw1fa7kaed2hpj30sg0l9q54.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
<img src="default.jpg" data-src="http://ww4.sinaimg.cn/large/006y8mN6gw1fa5obmqrmvj305k05k3yh.jpg" alt="">
</body>複製代碼
<script>
var num = document.getElementsByTagName('img').length;
var img = document.getElementsByTagName("img");
var n = 0; //存儲圖片加載到的位置,避免每次都從第一張圖片開始遍歷
lazyload(); //頁面載入完畢加載但是區域內的圖片
window.onscroll = lazyload;
function lazyload() { //監聽頁面滾動事件
var seeHeight = document.documentElement.clientHeight; //可見區域高度
var scrollTop = document.documentElement.scrollTop || document.body.scrollTop; //滾動條距離頂部高度
for (var i = n; i < num; i++) {
if (img[i].offsetTop < seeHeight + scrollTop) {
if (img[i].getAttribute("src") == "default.jpg") {
img[i].src = img[i].getAttribute("data-src");
}
n = i + 1;
}
}
}
</script>複製代碼
<script>
var n = 0,
imgNum = $("img").length,
img = $('img');
lazyload();
$(window).scroll(lazyload);
function lazyload(event) {
for (var i = n; i < imgNum; i++) {
if (img.eq(i).offset().top < parseInt($(window).height()) + parseInt($(window).scrollTop())) {
if (img.eq(i).attr("src") == "default.jpg") {
var src = img.eq(i).attr("data-src");
img.eq(i).attr("src", src);
n = i + 1;
}
}
}
}
</script>複製代碼
若是直接將函數綁定在scroll
事件上,當頁面滾動時,函數會被高頻觸發,這很是影響瀏覽器的性能。服務器
我想實現限制觸發頻率,來優化性能。app
節流函數:只容許一個函數在N秒內執行一次。下面是一個簡單的節流函數:函數
// 簡單的節流函數
//fun 要執行的函數
//delay 延遲
//time 在time時間內必須執行一次
function throttle(fun, delay, time) {
var timeout,
startTime = new Date();
return function() {
var context = this,
args = arguments,
curTime = new Date();
clearTimeout(timeout);
// 若是達到了規定的觸發時間間隔,觸發 handler
if (curTime - startTime >= time) {
fun.apply(context, args);
startTime = curTime;
// 沒達到觸發間隔,從新設定定時器
} else {
timeout = setTimeout(fun, delay);
}
};
};
// 實際想綁定在 scroll 事件上的 handler
function lazyload(event) {
for (var i = n; i < imgNum; i++) {
if (img.eq(i).offset().top < parseInt($(window).height()) + parseInt($(window).scrollTop())) {
if (img.eq(i).attr("src") == "default.jpg") {
var src = img.eq(i).attr("data-src");
img.eq(i).attr("src", src);
n = i + 1;
}
}
}
}
// 採用了節流函數
window.addEventListener('scroll',throttle(lazyload,500,1000));複製代碼