懶載入的3種實現方式
阿新 • • 發佈:2019-01-08
優勢
- 效能收益:瀏覽器載入圖片、decode、渲染都需要耗費資源,懶載入能節約效能消耗,縮短onload事件時間。
- 節約頻寬:這個不需要解釋。
通常,我們在html中展示圖片,會有兩種方式:
- img 標籤
- css background-image
img的懶載入實現
img有兩種方式實現懶載入:
- 事件監聽(scroll、resize、orientationChange)
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>event</title> <style> img { background: #F1F1FA; width: 400px; height: 300px; display: block; margin: 10px auto; border: 0; } </style> </head> <body> <img src="image1.jpg?tr=w-400,h-300" /> <img src="image2.jpg?tr=w-400,h-300" /> <img src="image3.jpg?tr=w-400,h-300" /> <img class="lazy" src="image2.jpg?tr=w-400,h-300" /> <img class="lazy" src="image3.jpg?tr=w-400,h-300" /> --> <img class="lazy" src="image4.jpg?tr=w-400,h-300" /> <img class="lazy" src="image5.jpg?tr=w-400,h-300" /> <img class="lazy" src="image6.jpg?tr=w-400,h-300" /> <img class="lazy" src="image7.jpg?tr=w-400,h-300" /> <img class="lazy" src="image8.jpg?tr=w-400,h-300" /> <img class="lazy" src="image9.jpg?tr=w-400,h-300" /> <img class="lazy" src="image10.jpg?tr=w-400,h-300" /> <script> document.addEventListener("DOMContentLoaded", function() { var lazyloadImages = document.querySelectorAll("img.lazy"); var lazyloadThrottleTimeout; function lazyload () { if(lazyloadThrottleTimeout) { clearTimeout(lazyloadThrottleTimeout); } lazyloadThrottleTimeout = setTimeout(function() { var scrollTop = window.pageYOffset; lazyloadImages.forEach(function(img) { if(img.offsetTop < (window.innerHeight + scrollTop)) { img.src = img.dataset.src; img.classList.remove('lazy'); } }); if(lazyloadImages.length == 0) { document.removeEventListener("scroll", lazyload); window.removeEventListener("resize", lazyload); window.removeEventListener("orientationChange", lazyload); } }, 20); } document.addEventListener("scroll", lazyload); window.addEventListener("resize", lazyload); window.addEventListener("orientationChange", lazyload); }); </script> </body> </html>
- Intersection Observer(相容性問題)
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>observer</title> <style> img { background: #F1F1FA; width: 400px; height: 300px; display: block; margin: 10px auto; border: 0; } </style> </head> <body> <img src="image2.jpg?tr=w-400,h-300" /> <img src="image3.jpg?tr=w-400,h-300" /> --> <img class="lazy" src="image1.jpg?tr=w-400,h-300" /> <img class="lazy" src="image2.jpg?tr=w-400,h-300" /> <img class="lazy" src="image3.jpg?tr=w-400,h-300" /> <img class="lazy" src="image4.jpg?tr=w-400,h-300" /> <img class="lazy" src="image5.jpg?tr=w-400,h-300" /> <img class="lazy" src="image6.jpg?tr=w-400,h-300" /> <img class="lazy" src="image7.jpg?tr=w-400,h-300" /> <img class="lazy" src="image8.jpg?tr=w-400,h-300" /> <img class="lazy" src="image9.jpg?tr=w-400,h-300" /> <img class="lazy" src="image10.jpg?tr=w-400,h-300" /> <script> document.addEventListener("DOMContentLoaded", function() { var lazyloadImages = document.querySelectorAll(".lazy"); var imageObserver = new IntersectionObserver(function(entries, observer) { entries.forEach(function(entry) { if (entry.isIntersecting) { var image = entry.target; image.src = image.dataset.src; image.classList.remove("lazy"); imageObserver.unobserve(image); } }); }); lazyloadImages.forEach(function(image) { imageObserver.observe(image); }); }); </script> </body> </html>
background-image的實現
background-image的實現跟img的原理基本是一樣的,區別是在對class的處理上:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>background</title> <style> body { margin: 0; } .bg { height: 200px; } #bg-image.lazy { background-image: none; background-color: #F1F1FA; } #bg-image { background-image: url("image1.jpg?tr=w-400,h-300"); background-size: 100%; } </style> </head> <body> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <div id="bg-image" class="bg lazy"></div> <script> document.addEventListener("DOMContentLoaded", function() { var lazyloadImages = document.querySelectorAll(".lazy"); var imageObserver = new IntersectionObserver(function(entries, observer) { entries.forEach(function(entry) { if (entry.isIntersecting) { var image = entry.target; image.classList.remove("lazy"); imageObserver.unobserve(image); } }); }); lazyloadImages.forEach(function(image) { imageObserver.observe(image); }); }); </script> </body> </html>
漸進式懶載入
漸進式懶載入,指的是存在降級處理,通常html形式如下:
<a href="full.jpg" class="progressive replace">
<img src="tiny.jpg" class="preview" alt="image" />
</a>
這樣的程式碼會有2個好處:
- 如果js執行失敗,可以點選預覽
- 大小與實際圖一致的佔位data URI,避免reflow
最終的程式碼如下:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>progressive</title>
<style>
a.progressive {
position: relative;
display: block;
overflow: hidden;
outline: none;
}
a.progressive:not(.replace) {
cursor: default;
}
a.progressive img {
display: block;
width: 100%;
max-width: none;
height: auto;
border: 0 none;
}
a.progressive img.preview {
filter: blur(2vw);
transform: scale(1.05);
}
a.progressive img.reveal {
position: absolute;
left: 0;
top: 0;
will-change: transform, opacity;
animation: reveal 1s ease-out;
}
@keyframes reveal {
0% {transform: scale(1.05); opacity: 0;}
100% {transform: scale(1); opacity: 1;}
}
</style>
</head>
<body>
<a href="nature5.jpg" data-srcset="https://s3-us-west-2.amazonaws.com/s.cdpn.io/123941/nature5.jpg 800w, nature5big.jpg 1600w" class="progressive replace">
<img src="" class="preview" alt="palm trees" />
</a>
<a href="nature2.jpg" class="progressive replace">
<img src="http://lorempixel.com/20/15/nature/2/" class="preview" alt="sunset" />
</a>
<a href="nature3.jpg" class="progressive replace">
<img src="http://lorempixel.com/20/15/nature/3/" class="preview" alt="tide" />
</a>
<a href="nature5.jpg" data-srcset="https://s3-us-west-2.amazonaws.com/s.cdpn.io/123941/nature5.jpg 800w, nature5big.jpg 1600w" class="progressive replace">
<img src="" class="preview" alt="palm trees" />
</a>
<a href="nature2.jpg" class="progressive replace">
<img src="http://lorempixel.com/20/15/nature/2/" class="preview" alt="sunset" />
</a>
<a href="nature3.jpg" class="progressive replace">
<img src="http://lorempixel.com/20/15/nature/3/" class="preview" alt="tide" />
</a>
<script>
window.addEventListener('load', function() {
var pItem = document.getElementsByClassName('progressive replace'), timer;
window.addEventListener('scroll', scroller, false);
window.addEventListener('resize', scroller, false);
inView();
function scroller(e) {
timer = timer || setTimeout(function() {
timer = null;
requestAnimationFrame(inView);
}, 300);
}
function inView() {
var scrollTop = window.pageYOffset;
var innerHeight = window.innerHeight;
var p = 0;
while (p < pItem.length) {
var offsetTop = pItem[p].offsetTop;
if (offsetTop < (scrollTop + innerHeight)) {
loadFullImage(pItem[p]);
pItem[p].classList.remove('replace');
}
else p++;
}
}
function loadFullImage(item) {
var img = new Image();
if (item.dataset) {
img.srcset = item.dataset.srcset || '';
img.sizes = item.dataset.sizes || '';
}
img.src = item.href;
img.className = 'reveal';
if (img.complete) addImg();
else img.onload = addImg;
function addImg() {
item.addEventListener('click', function(e) { e.preventDefault(); }, false);
item.appendChild(img).addEventListener('animationend', function(e) {
var pImg = item.querySelector('img.preview');
if (pImg) {
e.target.alt = pImg.alt || '';
item.removeChild(pImg);
e.target.classList.remove('reveal');
}
});
}
}
}, false);
</script>
</body>
</html>
現成庫
推薦下面這個庫,使用非常簡單:https://www.npmjs.com/package/lozad
https://segmentfault.com/a/1190000017795499來源: