記錄--兩行CSS讓頁面提升了近7倍渲染性能!

来源:https://www.cnblogs.com/smileZAZ/archive/2022/11/29/16936058.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 前言 對於前端人員來講,最令人頭疼的應該就是頁面性能了,當用戶在訪問一個頁面時,總是希望它能夠快速呈現在眼前並且是可交互狀態。如果頁面載入過慢,你的用戶很可能會因此離你而去。所以頁面性能對於前端開發者來說可謂是重中之重,其實你如果瞭解頁面 ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

前言

對於前端人員來講,最令人頭疼的應該就是頁面性能了,當用戶在訪問一個頁面時,總是希望它能夠快速呈現在眼前並且是可交互狀態。如果頁面載入過慢,你的用戶很可能會因此離你而去。所以頁面性能對於前端開發者來說可謂是重中之重,其實你如果瞭解頁面從載入到渲染完成的整個過程,就知道應該從哪方面下手了。

嗯,不要跑偏了,今天我們主要來研究長列表頁面的渲染性能

現如今的頁面越來越複雜,一個頁面往往承載著大量的元素,最常見的就是一些電商頁面,數以萬計的商品列表是怎麼保證渲染不卡頓的,大家在面對這種長列表渲染的場景下,一般都會採用分頁或者虛擬列表來減緩頁面一次性渲染的壓力,但這些方式都需要配合JS來時實現,那麼有沒有僅使用CSS就能夠實現的方案呢?

答案是有的,它就是我們今天的主角 —— 內容可見性(content-visibility)

如果這篇文章有幫助到你,❤️關註+點贊❤️鼓勵一下作者,文章公眾號首發,關註 前端南玖 第一時間獲取最新文章~

content-visibility

屬性值

content-visibility是CSS新增的屬性,主要用來提高頁面渲染性能,它可以控制一個元素是否渲染其內容,並且允許瀏覽器跳過這些元素的佈局與渲染。

  • visible:預設值,沒有效果。元素的內容被正常佈局和呈現。
  • hidden:元素跳過它的內容。跳過的內容不能被用戶代理功能訪問,例如在頁面中查找、標簽順序導航等,也不能被選擇或聚焦。這類似於給內容設置display: none
  • auto:該元素打開佈局包含、樣式包含和繪製包含。如果該元素與用戶不相關,它也會跳過其內容。與 hidden 不同,跳過的內容必須仍可正常用於用戶代理功能,例如在頁面中查找、tab 順序導航等,並且必須正常可聚焦和可選擇。

content-visibility: hidden手動管理可見性

上面說到content-visibility: hidden的效果與display: none類似,但其實兩者還是有比較大的區別的:

  • content-visibility: hidden 只是隱藏了子元素,自身不會被隱藏
  • content-visibility: hidden 隱藏內容的渲染狀態會被緩存,所以當它被移除或者設為可見時,瀏覽器不會重新渲染,而是會應用緩存,所以對於需要頻繁切換顯示隱藏的元素,這個屬性能夠極大地提高渲染性能。

content-v1.png

從這上面我們可以看到,添加了content-visibility: hidden元素的子元素確實是沒有渲染,但它自身是會渲染的!

content-visibility: auto 跳過渲染工作

我們仔細想想,頁面上雖然會有很多元素,但是它們會同時呈現在用戶眼前嗎,很顯然是不會的,用戶每次能夠真實看到就只有設備可見區那些內容,對於非可見區的內容只要頁面不發生滾動,用戶就永遠看不到。雖然用戶看不到,但瀏覽器卻會實實在在的去渲染,以至於浪費大量的性能。所以我們得想辦法讓瀏覽器不渲染非可視區的內容就能夠達到提高頁面渲染性能的效果。

我們上面說到的虛擬列表原理其實就跟這個類似,在首屏載入時,只載入可視區的內容,當頁面發生滾動時,動態通過計算獲得可視區的內容,並將非可視區的內容進行刪除,這樣就能夠大大提高長列表的渲染性能。

但這個需要配合JS才能實現,現在我們可以使用CSS中content-visibility: auto,它可以用來跳過屏幕外的內容渲染,對於這種有大量離屏內容的長列表,可以大大減少頁面渲染時間。

我們將上面的例子稍微改改:

<template>
  <div class="card_item">
    <div class="card_inner">
      <img :src="book.bookCover" class="book_cover" />
      <div class="card_item_right">
        <div class="book_title">{{ `${book.bookName}${index + 1}` }}</div>
        <div class="book_author">{{ book.catlog }}</div>
        <div class="book_tags">
          <div class="book_tag" v-for="(item, index) in book.tags" :key="index">
            {{ item }}
          </div>
        </div>
        <div class="book_desc">
          {{ book.desc }}
        </div>
      </div>
    </div>
  </div>
</template>

<script setup lang="ts">
import { toRefs } from "vue";

const props = defineProps<{
  book: any;
  index: any;
}>();
const { book, index } = toRefs(props);
</script>

<style lang="less" scoped>
.card_item {
  margin: 20px auto;
  content-visibility: auto;
}
  / *
  ...
  */
</style>

首先是沒有添加content-visibility: auto的效果,無論這些元素是否在可視區,都會被渲染

content-v2.png

如果我們在平常業務中這樣寫,用戶進入到這個頁面可能就直介面吐芬芳了,為了性能考慮,我們為每一個列表項加上:

.card_item {
  content-visibility: auto;
}

這個時候我們再來看下效果:

content-v3.png

從第10個開始,這些沒在可視區的元素就沒有被渲染,這可比上面那種全部元素都渲染好太多了,但是如果瀏覽器不渲染頁面內的一些元素,滾動將是一場噩夢,因為無法正確計算頁面高度。這是因為,content-visibility會將分配給它的元素的高度(height)視為0,瀏覽器在渲染之前會將這個元素的高度變為0,從而使我們的頁面高度和滾動變得混亂。

content-v4.gif

這裡我們可以看到頁面上的滾動條會出現抖動現象,這是因為可視區外的元素只有出現在了可視區才會被渲染,這就回導致前後頁面高度會發生變化,從而出現滾動條的詭異抖動現象,這是虛擬列表基本都會存在的問題。

⚠️註意:當元素接近視口時,瀏覽器不再添加size容器並開始繪製和命中測試元素的內容。這使得渲染工作能夠及時完成以供用戶查看。

這也是為什麼上面我們看到的是從第十個才開始不渲染子元素,因為它需要一個緩衝區以便瀏覽器能夠在頁面發生滾動時及時渲染呈現在用戶眼前。

上面提到的size其實是一種 CSS 屬性的潛在值contain,它指的是元素上的大小限制確保元素的框可以在不需要檢查其後代的情況下進行佈局。這意味著如果我們只需要元素的大小,我們可以跳過後代的佈局。

contain-intrinsic-size 救場

頁面在滾動過程中滾動條一直抖動,這是一個不能接受的體驗問題,為了更好地實現content-visibility,瀏覽器需要應用 size containment 以確保內容的渲染結果不會以任何方式影響元素的大小。這意味著該元素將像空的一樣佈局。如果元素沒有在常規塊佈局中指定的高度,那麼它將是 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 可以為元素指定以下一個或兩個值。如果指定了兩個值,則第一個值適用於寬度,第二個值適用於高度。如果指定單個值,則它適用於寬度和高度。

實現

我們只需要給添加了content-visibility: auto的元素添加上contain-intrinsic-size就能夠解決滾動條抖動的問題,當然,這個高度約接近真實渲染的高度,效果會越好,如果實在無法知道準確的高度,我們也可以給一個大概的值,也會使滾動條的問題相對減少。

.card_item {
  content-visibility: auto;
  contain-intrinsic-size: 200px;
}

content-v5.png

之前沒添加contain-intrinsic-size屬性時,可視區外的元素高度都是0,現在這些元素高度都是我們設置的contain-intrinsic-size的值,這樣的話整個頁面的高度就是不會發生變化(或者說變化很小),從而頁面滾動條也不會出現抖動問題(或者說抖動減少)

content-v6.gif

性能對比

上面說了這麼多,content-visibility是否真的能夠提高頁面的渲染性能呢,我們來實際對比看看:

  • 首先是沒有content-visibility的頁面渲染

content-v7.png

  • 然後是有content-visibility的頁面渲染

content-v8.png

上面是用1000個列表元素進行測試的,有content-visibility的頁面渲染花費時間大概是37ms,而沒有content-visibility的頁面渲染花費時間大概是269ms,提升了足足有7倍之多!!!

對於列表元素更多的頁面,content-visibility帶來的渲染性能提升會更加明顯。

思考

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • public static void GetRegistData() { string name = "huishuangzhu"; //搜索到註冊表根目錄 RegistryKey hkml = Registry.ClassesRoot; //搜索到註冊表根目錄下的XXX文件夾。 RegistryK ...
  • 溫馨提示,請使用ctrl+F進行快速查找 ws2_32.lib error LNK2001: 無法解析的外部符號 __imp_htons error LNK2001: 無法解析的外部符號 __imp_ntohl error LNK2001: 無法解析的外部符號 __imp_ntohs error L ...
  • 1、什麼是MQTT? MQTT(Message Queuing Telemetry Transport,消息隊列遙測傳輸協議),是一種基於發佈/訂閱(publish/subscribe)模式的"輕量級"通訊協議,該協議構建於TCP/IP協議上,由IBM在1999年發佈。MQTT最大優點在於,可以以極 ...
  • 1.玻利維亞 MOPSV 為 5G 移動服務分配 3.3-3.6 GHz 頻段 https://www.oopp.gob.bo/wp-content/uploads/2022/10/2022-RM-174-Modificacion-al-Plan-Nacional-de-Frecuencia.pdf ...
  • 視圖 create view ... as ps:SQL文件在上一篇博客末尾 視圖就是通過查詢得到一張虛擬表,然後保存下來,下次直接使用 create view teacher_course as select * from teacher inner join course on teacher. ...
  • 10.1 事務的基本概念: 什麼是事務?事務是用戶定義的一個資料庫操作序列,該操作要麼全做,要麼全不做,是一個不可分割的工作單位,是恢復(知識點)和併發控制(知識點)的基本單位 事務和程式的區別: 在關係資料庫中,一個事務可以是一條SQL語句,或多條SQL語句,或整個程式 一個程式可以有多個事務 事 ...
  • 本篇開啟資料庫在工作中常用到的格式轉換與工具,歡迎大家評論留言:smile: SQL將小數轉為保留兩位的百分數 CONCAT(CONVERT((<需要轉換的值>)*100,DECIMAL(18,2)),'%') turnNum 常用的日期格式化 引用的是CSDN博主isTrueLoveColour的 ...
  • 案例介紹 歡迎來到我的小院,我是霍大俠,恭喜你今天又要進步一點點了!我們來用JavaScript相關知識,做一個隨機點名的案例。你可以通過點擊開始按鈕控制上方名字的閃動,點擊停止按鈕可以隨機選定一個名字。 案例演示 運行程式後,我們可以看到一個矩形框按鈕,顯示開始點名,點擊後名字隨機閃動。同時按鈕變 ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...