前端(vue)入門到精通課程:進入學習
Apipost = Postman + Swagger + Mock + Jmeter 超好用的API調試工具:點擊使用
對于前端人員來講,最令人頭疼的應該就是頁面性能了,當用戶在訪問一個頁面時,總是希望它能夠快速呈現(xiàn)在眼前并且是可交互狀態(tài)。如果頁面加載過慢,你的用戶很可能會因此離你而去。所以頁面性能對于前端開發(fā)者來說可謂是重中之重,其實你如果了解頁面從加載到渲染完成的整個過程,就知道應該從哪方面下手了。
嗯,不要跑偏了,今天我們主要來研究長列表頁面的渲染性能
現(xiàn)如今的頁面越來越復雜,一個頁面往往承載著大量的元素,最常見的就是一些電商頁面,數以萬計的商品列表是怎么保證渲染不卡頓的,大家在面對這種長列表渲染的場景下,一般都會采用分頁或者虛擬列表來減緩頁面一次性渲染的壓力,但這些方式都需要配合JS來時實現(xiàn),那么有沒有僅使用CSS就能夠實現(xiàn)的方案呢?
答案是有的,它就是我們今天的主角 —— 內容可見性(content-visibility)?!就扑]學習:css視頻教程】
content-visibility
屬性值
content-visibility
是CSS新增的屬性,主要用來提高頁面渲染性能,它可以控制一個元素是否渲染其內容,并且允許瀏覽器跳過這些元素的布局與渲染。
- visible:默認值,沒有效果。元素的內容被正常布局和呈現(xiàn)。
- hidden:元素跳過它的內容。跳過的內容不能被用戶代理功能訪問,例如在頁面中查找、標簽順序導航等,也不能被選擇或聚焦。這類似于給內容設置
display: none
。 - auto:該元素打開布局包含、樣式包含和繪制包含。如果該元素與用戶不相關,它也會跳過其內容。與 hidden 不同,跳過的內容必須仍可正常用于用戶代理功能,例如在頁面中查找、tab 順序導航等,并且必須正??删劢购涂蛇x擇。
content-visibility: hidden手動管理可見性
上面說到content-visibility: hidden
的效果與display: none
類似,但其實兩者還是有比較大的區(qū)別的:
- content-visibility: hidden 只是隱藏了子元素,自身不會被隱藏
- content-visibility: hidden 隱藏內容的渲染狀態(tài)會被緩存,所以當它被移除或者設為可見時,瀏覽器不會重新渲染,而是會應用緩存,所以對于需要頻繁切換顯示隱藏的元素,這個屬性能夠極大地提高渲染性能。
從這上面我們可以看到,添加了content-visibility: hidden
元素的子元素確實是沒有渲染,但它自身是會渲染的!
content-visibility: auto 跳過渲染工作
我們仔細想想,頁面上雖然會有很多元素,但是它們會同時呈現(xiàn)在用戶眼前嗎,很顯然是不會的,用戶每次能夠真實看到就只有設備可見區(qū)那些內容,對于非可見區(qū)的內容只要頁面不發(fā)生滾動,用戶就永遠看不到。雖然用戶看不到,但瀏覽器卻會實實在在的去渲染,以至于浪費大量的性能。所以我們得想辦法讓瀏覽器不渲染非可視區(qū)的內容就能夠達到提高頁面渲染性能的效果。
我們上面說到的虛擬列表原理其實就跟這個類似,在首屏加載時,只加載可視區(qū)
的內容,當頁面發(fā)生滾動時,動態(tài)通過計算獲得可視區(qū)
的內容,并將非可視區(qū)
的內容進行刪除,這樣就能夠大大提高長列表的渲染性能。
但這個需要配合JS才能實現(xiàn),現(xiàn)在我們可以使用CSS中content-visibility: auto
,它可以用來跳過屏幕外的內容渲染,對于這種有大量離屏內容的長列表,可以大大減少頁面渲染時間。
我們將上面的例子稍微改改:
<template> <div> <div> <img :src="book.bookCover" /> <div> <div>{{ `${book.bookName}${index + 1}` }}</div> <div>{{ book.catlog }}</div> <div> <div v-for="(item, index) in book.tags" :key="index"> {{ item }} </div> </div> <div> {{ book.desc }} </div> </div> </div> </div> </template> <script setup> import { toRefs } from "vue"; const props = defineProps<{ book: any; index: any; }>(); const { book, index } = toRefs(props); </script> <style scoped> .card_item { margin: 20px auto; content-visibility: auto; } / * ... */ </style>
首先是沒有添加content-visibility: auto
的效果,無論這些元素是否在可視區(qū),都會被渲染
如果我們在平常業(yè)務中這樣寫,用戶進入到這個頁面可能就直接口吐芬芳了,為了性能考慮,我們?yōu)槊恳粋€列表項加上:
.card_item { content-visibility: auto; }
這個時候我們再來看下效果:
從第10個開始,這些沒在可視區(qū)的元素就沒有被渲染,這可比上面那種全部元素都渲染好太多了,但是如果瀏覽器不渲染頁面內的一些元素,滾動將是一場噩夢,因為無法正確計算頁面高度。這是因為,content-visibility
會將分配給它的元素的高度(height
)視為0
,瀏覽器在渲染之前會將這個元素的高度變?yōu)?code>0,從而使我們的頁面高度和滾動變得混亂。
這里我們可以看到頁面上的滾動條會出現(xiàn)抖動現(xiàn)象,這是因為可視區(qū)外的元素只有出現(xiàn)在了可視區(qū)才會被渲染,這就回導致前后頁面高度會發(fā)生變化,從而出現(xiàn)滾動條的詭異抖動現(xiàn)象,這是虛擬列表基本都會存在的問題。
⚠️注意:當元素接近視口時,瀏覽器不再添加size
容器并開始繪制和命中測試元素的內容。這使得渲染工作能夠及時完成以供用戶查看。
這也是為什么上面我們看到的是從第十個才開始不渲染子元素,因為它需要一個緩沖區(qū)以便瀏覽器能夠在頁面發(fā)生滾動時及時渲染呈現(xiàn)在用戶眼前。
上面提到的size
其實是一種 CSS 屬性的潛在值contain
,它指的是元素上的大小限制確保元素的框可以在不需要檢查其后代的情況下進行布局。這意味著如果我們只需要元素的大小,我們可以跳過后代的布局。
contain-intrinsic-size 救場
頁面在滾動過程中滾動條一直抖動,這是一個不能接受的體驗問題,為了更好地實現(xiàn)content-visibility
,瀏覽器需要應用 size containment 以確保內容的渲染結果不會以任何方式影響元素的大小。這意味著該元素將像空的一樣布局。如果元素沒有在常規(guī)塊布局中指定的高度,那么它將是 0 高度。
這個時候我們可以使用contain-intrinsic-size
來指定的元素自然大小,確保我們未渲染子元素的 div 仍然占據空間,同時也保留延遲渲染的好處。
語法
此屬性是以下 CSS 屬性的簡寫:
contain-intrinsic-width
contain-intrinsic-height
/* Keyword values */ contain-intrinsic-width: none; /* <length> values */ contain-intrinsic-size: 1000px; contain-intrinsic-size: 10rem; /* width | height */ contain-intrinsic-size: 1000px 1.5em; /* auto <length> */ contain-intrinsic-size: auto 300px; /* auto width | auto height */ contain-intrinsic-size: auto 300px auto 4rem;
contain-intrinsic-size 可以為元素指定以下一個或兩個值。如果指定了兩個值,則第一個值適用于寬度,第二個值適用于高度。如果指定單個值,則它適用于寬度和高度。
實現(xiàn)
我們只需要給添加了content-visibility: auto
的元素添加上contain-intrinsic-size
就能夠解決滾動條抖動的問題,當然,這個高度約接近真實渲染的高度,效果會越好,如果實在無法知道準確的高度,我們也可以給一個大概的值,也會使?jié)L動條的問題相對減少。
.card_item { content-visibility: auto; contain-intrinsic-size: 200px; }
之前沒添加contain-intrinsic-size
屬性時,可視區(qū)外的元素高度都是0,現(xiàn)在這些元素高度都是我們設置的contain-intrinsic-size
的值,這樣的話整個頁面的高度就是不會發(fā)生變化(或者說變化很?。瑥亩撁鏉L動條也不會出現(xiàn)抖動問題(或者說抖動減少)
性能對比
上面說了這么多,content-visibility
是否真的能夠提高頁面的渲染性能呢,我們來實際對比看看:
- 首先是沒有
content-visibility
的頁面渲染
- 然后是有
content-visibility
的頁面渲染
上面是用1000個列表元素進行測試的,有content-visibility
的頁面渲染花費時間大概是37ms,而沒有content-visibility
的頁面渲染花費時間大概是269ms,提升了足足有7倍之多!??!
對于列表元素