地址:前端面试题库

最近开发了一个功能,页面首页会加载大量的图片,初次进入页面时,会导致页面性能下降,
于是乎,我改进了这个功能,可以让所有图片自动懒加载。
这个功能主要的底层逻辑是是使用IntersectionObserver API,IntersectionObserver用于在浏览器中观察元素的可见性和位置变化。它可以帮助开发者实现一些动态行为,如图片的懒加载、无限滚动等。
简单的示例如下:
- // 创建IntersectionObserver实例
- const observer = new IntersectionObserver((entries, observer) => {
- // 遍历观察的元素
- entries.forEach(entry => {
- // 如果元素可见
- if (entry.isIntersecting) {
- // 加载图片
- const img = entry.target;
- const src = img.getAttribute('data-src');
- img.setAttribute('src', src);
- // 停止观察该元素
- observer.unobserve(img);
- }
- });
- });
-
- // 获取所有需要懒加载的图片元素
- const lazyImages = document.querySelectorAll('.lazy-image');
-
- // 观察每个图片元素
- lazyImages.forEach(image => {
- observer.observe(image);
- });
接下来我们实现一个通用的 hook,基本的功能如下:
src,同时提供data-src属性ref 属性。data-src属性替换 src 属性- import { onMounted, Ref } from "vue";
- const options = {
- // root: document.querySelector(".container"), // 根元素,默认为视口
- rootMargin: "0px", // 根元素的边距
- threshold: 0.5, // 可见性比例阈值
- once: true,
- };
-
- function callback(
- entries: IntersectionObserverEntry[],
- observer: IntersectionObserver
- ) {
- entries.forEach((entry) => {
- // 处理每个目标元素的可见性变化
- if (entry.intersectionRatio <= 0) return;
- const img: Element = entry.target;
- const src = img.getAttribute("data-src");
-
- img.setAttribute("src", src ?? ""); // 将真实的图片地址赋给 src 属性
-
- observer.unobserve(img);
- });
- }
-
- export const useInView = (ref: Ref) => {
- const observer = new IntersectionObserver(callback, options);
-
- onMounted(() => {
- Object.keys(ref.value).forEach((e) => observer.observe(ref.value[e]));
- });
- };
-
-
- <template>
- <h4>公众号:萌萌哒草头将军h4>
- <div
- v-for="(_, idx) in new Array(200).fill(11)"
- >
- <img
- ref="imgRef"
- src="https://via.placeholder.com/200"
- :data-src="`https://picsum.photos/200/${180 + idx}`"
- alt="b"
- />
- div>
- template>
实际效果如下

虽然基本的功能要求已经完成了,但是现在还不够优雅!!!
接下来,我们增加个过渡动画。每次当加载完图片,就从占位图过渡到正常图片模式。
- img.onload = () => {
- img.setAttribute('class', 'fade-in')
- }
- @keyframes fadeIn {
- from {
- opacity: 0;
- }
- to {
- opacity: 1;
- }
- }
-
- /* 应用淡入动画到元素 */
- .fade-in {
- animation: fadeIn 0.6s ease-in;
- }

完整代码如下:
- import { onMounted, Ref } from "vue";
- const options = {
- // root: document.querySelector(".container"), // 根元素,默认为视口
- rootMargin: "0px", // 根元素的边距
- threshold: 0.5, // 可见性比例阈值
- once: true,
- };
-
- function callback(
- entries: IntersectionObserverEntry[],
- observer: IntersectionObserver
- ) {
- entries.forEach((entry) => {
- if (entry.intersectionRatio <= 0) return;
- const img = entry.target as HTMLImageElement;
- const src = img.getAttribute("data-src");
-
- img.setAttribute("src", src ?? ""); // 将真实的图片地址赋给 src 属性
-
- img.onload = () => {
- img.setAttribute("class", "fade-in");
- };
-
- observer.unobserve(img);
- });
- }
-
- export const useInView = (ref: Ref) => {
- const observer = new IntersectionObserver(
- callback,
- options
- );
-
- onMounted(() => {
- Object.keys(ref.value)
- .forEach((e) => observer.observe(ref.value[e]));
- });
- };
-
- <template>
- <h4>公众号:萌萌哒草头将军h4>
- <div
- v-for="(_, idx) in new Array(200).fill(11)"
- style="width: 200px height: 200px;"
- >
- <img
- ref="imgRef"
- style="height: 100%"
- src="https://via.placeholder.com/200"
- :data-src="`https://picsum.photos/200/${180 + idx}`"
- alt="b"
- />
- div>
- template>
-
- <style scoped>
- /* 定义淡入动画 */
- @keyframes fadeIn {
- from {
- opacity: 0;
- }
- to {
- opacity: 1;
- }
- }
-
- /* 应用淡入动画到元素 */
- .fade-in {
- animation: fadeIn 0.6s ease-in;
- }
- style>
地址:前端面试题库
