Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
Tags
- vue.js
- 개발콘텐츠
- click and drag
- TSDoc
- React.js
- reactjs
- 커스텀
- 레이아웃쪼개기
- typescript
- JS console
- 제네릭
- 누구나 자료구조와 알고리즘
- 공통컴포넌트
- 타입스크립트
- javascript
- 2022
- 타입좁히기
- utilty type
- 리액트
- 성능최적화
- returnType
- 반복줄이기
- Chart.js
- NonNullable
- const 단언문
- react
- 폰트적용하기
- 티스토리꾸미기
- CSS
- React Native
Archives
- Today
- Total
몽땅뚝딱 개발자
성능 최적화하기 - (7) Element Lazy Loading 본문
◽ 문제상황
동영상이 메인컨텐츠이지만 이미지가 먼저 다운로드되고 여유가 있을 때 동영상이 다운로드 된다.
따라서 그 전에는 pending 된 상태로 렌더링이 되지 않은 화면이 사용자에게 보여진다.
아래와 같은 해결책이 있다.
1. 이미지를 빠르게 다운로드한다. => 하지만 궁극적인 해결방법이 아니다.
2. ⭐️ lazy loading을 사용한다.
◽ lazy loading이란?
element가 화면에 보여져야 할 타이밍에 element를 그린다.
이미지가 보여지는 부분까지 스크롤이 되었느냐 되지않았느냐로 판단하기때문에 스크롤 이벤트를 사용한다.
◽ Intersection Observer(=관찰자)의 사용
첫번째 인자는 콜백함수, 두번째 인자는 옵션을 사용한다.
boxElement는 Dom Element로 Observer에 관찰되는 상태가 되며 화면에 해당 element가 보일 때 실행된다.
[예시]
function createObserver() {
let observer;
let options = {
root: null,
rootMargin: "0px",
threshold: buildThresholdList()
};
observer = new IntersectionObserver(handleIntersect, options);
observer.observe(boxElement);
}
[종류]
// entry.boundingClientRect
// entry.intersectionRatio
// entry.intersectionRect
// entry.isIntersecting
// entry.rootBounds
// entry.target
// entry.time
[적용 예제]
import React, {useEffect, useRef} from 'react'
function Card(props) {
const imgRef = useRef(null)
// 최초에 한 번만 실행하기 위해 useEffect를 사용한다.
useEffect(() => {
const callback = (entries, observer) => {
// entries: observer 객체들을 담은 배열
// 아래의 경우처럼 총 3번 실행된다.
// 1. 지정한 엘레멘트 화면에 보이는 순간
// 2. 지정한 엘레멘트 화면에 안 보이는 순간
// 3. 옵저버 객체 생성 되는 순간
entries.forEach(entry => {
if(entry.isIntersecting) {
entry.target.src = entry.target.dataset.src
// 'unobserver'를 사용하여 최초 한번만 실행되도록 한다.
observer.unobserve(entry.target)
}
});
}
const options = {}
const observer = new IntersectionObserver(callback, options)
observer.observe(imgRef.current)
}, [])
return (
<div className="Card text-center">
<img data-src={props.image} ref={imgRef}/>
<div className="p-5 font-semibold text-gray-700 text-xl md:text-lg lg:text-xl keep-all">
{props.children}
</div>
</div>
)
}
export default Card
출처
'Development > 성능최적화' 카테고리의 다른 글
성능 최적화하기 - (9) 폰트 최적화 (0) | 2022.06.02 |
---|---|
성능 최적화하기 - (8) 이미지, 동영상 최적화 (0) | 2022.05.31 |
성능 최적화하기 - (6) 코드 스플리팅(Code Spliting) (0) | 2022.05.27 |
성능 최적화하기 - (5) 프리로딩(Preloading) (0) | 2022.05.27 |
성능 최적화하기 - (4) Component Lazy Loading (0) | 2022.05.27 |
Comments