欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

antd?3.x?Table組件如何快速實(shí)現(xiàn)虛擬列表詳析

 更新時(shí)間:2022年11月29日 09:36:08   作者:Yue櫟廷  
這篇文章主要給大家介紹了關(guān)于antd?3.x?Table組件如何快速實(shí)現(xiàn)虛擬列表的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用antd具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

1. 前言

隨著互聯(lián)網(wǎng)的發(fā)展,web展示的內(nèi)容越來(lái)越豐富,也越來(lái)越無(wú)窮。我們?cè)趯?shí)際開(kāi)發(fā)中難免會(huì)遇到長(zhǎng)列表數(shù)據(jù)渲染,而又不適合分頁(yè)的業(yè)務(wù)場(chǎng)景,如果瀏覽器直接渲染海量數(shù)據(jù),會(huì)造成頁(yè)面卡死,嚴(yán)重時(shí)導(dǎo)致瀏覽器資源耗盡,直接崩潰掉。這種情況用戶與產(chǎn)品是無(wú)法接受的,瀏覽器性能與業(yè)務(wù)需求產(chǎn)生了對(duì)立,因此虛擬列表技術(shù)被提出,為這種尷尬的場(chǎng)面提供了一線生機(jī)。

2. 虛擬列表

虛擬列表其實(shí)是按需顯示的一種實(shí)現(xiàn),即只對(duì)可見(jiàn)區(qū)域進(jìn)行渲染,對(duì)非可見(jiàn)區(qū)域中的數(shù)據(jù)不渲染或部分渲染的技術(shù),從而達(dá)到極高的渲染性能。假設(shè)有10萬(wàn)條記錄需要同時(shí)渲染,我們屏幕的可見(jiàn)區(qū)域的高度為500px,而列表項(xiàng)的高度為50px,則此時(shí)我們?cè)谄聊恢凶疃嘀荒芸吹?0個(gè)列表項(xiàng),那么在渲染的時(shí)候,我們只需加載可視區(qū)的那10條即可,觸發(fā)頁(yè)面滾動(dòng)時(shí),實(shí)時(shí)替換當(dāng)前應(yīng)該展示在頁(yè)面中的10條數(shù)據(jù)。

它的名詞解釋由一張圖來(lái)詮釋,如下:

觸發(fā)滾動(dòng)后,可視區(qū)域內(nèi)的數(shù)據(jù)變化:

  • 首先,定義一個(gè)visibleHeight變量來(lái)保存我們可見(jiàn)區(qū)域的高度,作為內(nèi)容容器,并設(shè)置為500px,visibleHeight = 500
  • 假設(shè)每列高度itemHeight固定,則可見(jiàn)區(qū)域內(nèi)的數(shù)據(jù)條數(shù)visibleCount = Math.ceil(visibleHeight / itemHeight)。
  • 由于只渲染可視區(qū)域內(nèi)的數(shù)據(jù),所以我們需要另一個(gè)占位容器,使父盒子出現(xiàn)滾動(dòng)條,占位容器高度placeholderHeight = totalCount * itemHeight。占位容器替代了內(nèi)容容器,撐出了滾動(dòng)條,內(nèi)容盒子則采用絕對(duì)定位脫離文檔流。
  • 每當(dāng)觸發(fā)滾動(dòng)時(shí),獲取滾動(dòng)條的滾動(dòng)距離,計(jì)算出應(yīng)該總共滾動(dòng)了的數(shù)據(jù)條數(shù),從而設(shè)置數(shù)據(jù)的開(kāi)始索引startIdx = Math.floor(scrollTop / itemHeight),而結(jié)束索引endIdx = startIdx + visibleCount。
  • 如果是原生開(kāi)發(fā),則根據(jù)開(kāi)始、結(jié)束索引去操作dom,替換dom。如果是vue或react都是數(shù)據(jù)驅(qū)動(dòng),則更新要渲染的list數(shù)據(jù)即可,renderList = sourceList.slice(startIdx, endIdx)
  • 最后一點(diǎn),我們的可視數(shù)據(jù)列表需要根據(jù)每次滾動(dòng)的距離相應(yīng)地調(diào)整內(nèi)容容器的位置,以保證在父盒子的滾動(dòng)下,內(nèi)容容器始終在可視區(qū)域中,offset = startIdx * itemCount,offset則為內(nèi)容容器相對(duì)于占位容器的偏移距離。

首先,準(zhǔn)備dom結(jié)構(gòu):

<div className="wrapper" style={{
    position: "relative",
    overflow: "auto"
}}>
    <div className="placeholder-list" style={{ 
        height: `${visibleHeight}px` 
    }}></div>
    <div className="render-list" style={{ 
        postion: "absolute", 
        top: 0, 
        left: 0
    }}>...</div>
</div>

wrapper添加滾動(dòng)事件實(shí)現(xiàn)邏輯:

scrollEvent(e){
    const startIdx = Math.floor(e.target.scrollTop / itemHeight);
    const endIdx = startIdx + visibleCount;
    setList(source.slice(startIdx, endIdx));
    // 設(shè)置偏移距離,保持?jǐn)?shù)據(jù)在視圖中
    const offset = startIdx * itemHeight;
    listRef.current.style.top = offset + "px";
}

我們發(fā)現(xiàn),快速滾動(dòng)時(shí)最下方會(huì)出現(xiàn)空白的現(xiàn)象,因?yàn)榇藭r(shí)數(shù)據(jù)還沒(méi)渲染成功。為了優(yōu)化此空白,考慮多渲染2條數(shù)據(jù)作為緩沖區(qū)。因此visibelCount=Math.ceil(visibelHeight / itemHeight) + 2

代碼完整示例:

import { useCallback, useEffect, useRef, useState } from "react";

const visibleHeight = 360;
const itemHeight = 50;
const visibleCount = Math.ceil(visibleHeight / itemHeight) + 2;
const totalCount = 100;

const source = Array.from(Array(totalCount), (item, index) => index);

export default function VirtualList() {
  const [list, setList] = useState(source);
  const listRef = useRef();

  const scrollEvent = useCallback((e) => {
    const startIdx = Math.floor(e.target.scrollTop / itemHeight);
    const endIdx = startIdx + visibleCount;
    setList(source.slice(startIdx, endIdx));
    const offset = startIdx * itemHeight;
    listRef.current.style.top = offset + "px";
  }, []);
  
  useEffect(() => {
    listRef.current = document.querySelector(".list");
  }, []);
  
  return (
    <div
      style={{
        backgroundColor: "#FFF",
        height: visibleHeight + 'px',
        textAlign: "center",
        overflow: "auto",
        position: "relative",
        overscrollBehavior: 'contain'
      }}
      onScroll={scrollEvent}
    >
      <div style={{ height: totalCount * itemHeight + 'px' }}></div>
      <div
        className="list"
        style={{
          position: "absolute",
          top: 0,
          left: 0,
          width: "100%",
          height: visibleHeight + 'px'
        }}
      >
        {list.map((item) => {
          return (
            <div
              key={item}
              style={{ height: itemHeight + 'px', borderBottom: "1px solid #eee" }}
            >
              {item}
            </div>
          );
        })}
      </div>
    </div>
  );
}

3. 虛擬table

終于來(lái)到了標(biāo)題的內(nèi)容,如何對(duì)antd table3.x進(jìn)行虛擬表格的封裝。其實(shí)和上述的代碼差不多,只不過(guò)對(duì)于有的新手同學(xué)來(lái)講,可能有點(diǎn)摸不著入口,所以有了本節(jié)內(nèi)容。

目前在antd4.x版本table已經(jīng)實(shí)現(xiàn)了開(kāi)啟虛擬列表的配置,拿來(lái)即用。針對(duì)3.x的版本自己實(shí)現(xiàn)了一個(gè)虛擬table,解決了業(yè)務(wù)上長(zhǎng)列表渲染卡頓的問(wèn)題。

注意:Table每項(xiàng)需要定高,因此columns屬性中需要ellipsis:true保證數(shù)據(jù)只展示一行,溢出展示省略號(hào)。

根據(jù)上節(jié)內(nèi)容介紹的虛擬列表思路,我們需要準(zhǔn)備2個(gè)容器,一個(gè)內(nèi)容容器,Table已經(jīng)提供了,另一個(gè)占位容器沒(méi)有提供,所以需要手動(dòng)創(chuàng)建一個(gè)并放在合適的地方。通過(guò)開(kāi)發(fā)者工具審查元素找到Table內(nèi)提供的那個(gè)內(nèi)容容器.ant-table-body table,獲取其dom。父容器.ant-table-body,創(chuàng)建一個(gè)占位容器div,追加到父容器內(nèi)。通過(guò)元素審查也知道Table tr高度為54px,即itemHeight=54

知道了類名,就可以獲取到Table的dom為所欲為了。

useEffect(() => {
    const parentNode = document.querySelector('.ant-table-body');
    const table = document.querySelector('.ant-table-body table');
    // 用ref保持table方便在滾動(dòng)事件中使用table dom
    tableRef.current = table;
    // 創(chuàng)建一個(gè)占位的div,高度等于所有數(shù)據(jù)高度,用來(lái)?yè)伍_(kāi)容器展示滾動(dòng)條
    const placeholderWrapper = document.createElement('div');
    placeholderWrapper.style.height = itemHeight * totalCount + 'px'
    parentNode.appendChild(placeholderWrapper);
    // 子絕父相口訣,為table設(shè)置定位,脫離文檔流,把位置讓給占位盒子
    parentNode.style.position = 'relative';
    table.style.position = 'absolute';
    table.style.top = 0;
    table.style.left = 0;
    // 添加滾動(dòng)事件
    parentNode.addEventListener('scroll', scrollEvent)
    return () => {
      // 清理占位盒子
      parentNode.removeChild(placeholderWrapper);
      parentNode.removeEventListener('scroll', scrollEvent)
    }
  }, [scrollEvent]);

接下來(lái)實(shí)現(xiàn)滾動(dòng)事件,和上節(jié)內(nèi)容一致,保存范圍索引到state中:

const scrollEvent = useCallback((e) => {
    const startIdx = Math.floor(e.target.scrollTop / itemHeight);
    const endIdx = startIdx + visibleCount;
    // 保存當(dāng)前的范圍索引,用來(lái)slice源數(shù)據(jù)給展示用
    setRange([startIdx, endIdx]);
    const offset = startIdx * itemHeight;
    tableRef.current.style.top = offset + "px";
  }, []);

根據(jù)范圍索引,截取當(dāng)前要展示的數(shù)據(jù)項(xiàng)

const [range, setRange] = useState([]);
// 這個(gè)renderList就是需要給Table組件的
const renderList = useMemo(() => {
    const [start, end] = range;
    return dataSource.slice(start, end)
  }, [range])

return <Table dataSource={renderList} />

全文示例代碼Github地址。

4.總結(jié)

本文只是實(shí)現(xiàn)了在固定每項(xiàng)列表高度的情況下的虛擬列表,現(xiàn)實(shí)很多情況是不定高的。這個(gè)比定高的復(fù)雜,不過(guò)原理也是一樣的,多了一步需要計(jì)算渲染后的實(shí)際高度的步驟。后續(xù)會(huì)完善不定高的虛擬列表的實(shí)現(xiàn)。

本文的內(nèi)容也是我在工作中遇到的情況,應(yīng)該很多其他小伙伴也會(huì)遇到antd 3.x table的虛擬化的問(wèn)題,希望能給小伙伴們一點(diǎn)思路。因此有了本文,也是自己一次關(guān)于輸入與輸出的記錄與沉淀。

到此這篇關(guān)于antd 3.x Table組件如何快速實(shí)現(xiàn)虛擬列表的文章就介紹到這了,更多相關(guān)antd 3.x Table組件虛擬列表內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • ReactJS?應(yīng)用兼容ios9對(duì)標(biāo)ie11解決方案

    ReactJS?應(yīng)用兼容ios9對(duì)標(biāo)ie11解決方案

    這篇文章主要為大家介紹了ReactJS?應(yīng)用兼容ios9對(duì)標(biāo)ie11解決方案詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01
  • 工程級(jí)?React?注冊(cè)登錄全棧級(jí)流程分析

    工程級(jí)?React?注冊(cè)登錄全棧級(jí)流程分析

    這篇文章主要介紹了工程級(jí)?React?注冊(cè)登錄全棧級(jí)流程,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-02-02
  • React組件的解耦技巧分享

    React組件的解耦技巧分享

    本文我們將和大家一起來(lái)研究如何有效地將組件解耦,讓我們的代碼變的復(fù)用性極高,文中通過(guò)代碼示例講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2023-11-11
  • 深入學(xué)習(xí)TypeScript 、React、 Redux和Ant-Design的最佳實(shí)踐

    深入學(xué)習(xí)TypeScript 、React、 Redux和Ant-Design的最佳實(shí)踐

    這篇文章主要介紹了深入學(xué)習(xí)TypeScript 、React、 Redux和Ant-Design的最佳實(shí)踐,TypeScript 增加了代碼的可讀性和可維護(hù)性,擁有活躍的社區(qū),,需要的朋友可以參考下
    2019-06-06
  • React高級(jí)特性Context萬(wàn)字詳細(xì)解讀

    React高級(jí)特性Context萬(wàn)字詳細(xì)解讀

    React的context就是一個(gè)全局變量,可以從根組件跨級(jí)別在React的組件中傳遞。React context的API有兩個(gè)版本,React16.x之前的是老版本的context,之后的是新版本的context
    2022-11-11
  • 詳解create-react-app 2.0版本如何啟用裝飾器語(yǔ)法

    詳解create-react-app 2.0版本如何啟用裝飾器語(yǔ)法

    這篇文章主要介紹了詳解create-react-app 2.0版本如何啟用裝飾器語(yǔ)法,cra2.0時(shí)代如何啟用裝飾器語(yǔ)法呢? 我們依舊采用的是react-app-rewired, 通過(guò)劫持webpack cofig對(duì)象, 達(dá)到修改的目的
    2018-10-10
  • 關(guān)于useEffect執(zhí)行兩次的問(wèn)題及解決

    關(guān)于useEffect執(zhí)行兩次的問(wèn)題及解決

    這篇文章主要介紹了關(guān)于useEffect執(zhí)行兩次的問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-09-09
  • React組件實(shí)例三大核心屬性State props Refs詳解

    React組件實(shí)例三大核心屬性State props Refs詳解

    組件實(shí)例的三大核心屬性是:State、Props、Refs。類組件中這三大屬性都存在。函數(shù)式組件中訪問(wèn)不到 this,也就不存在組件實(shí)例這種說(shuō)法,但由于它的特殊性(函數(shù)可以接收參數(shù)),所以存在Props這種屬性
    2022-12-12
  • 解決React報(bào)錯(cuò)Encountered?two?children?with?the?same?key

    解決React報(bào)錯(cuò)Encountered?two?children?with?the?same?key

    這篇文章主要為大家介紹了React報(bào)錯(cuò)Encountered?two?children?with?the?same?key解決方法,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • react源碼中的生命周期和事件系統(tǒng)實(shí)例解析

    react源碼中的生命周期和事件系統(tǒng)實(shí)例解析

    這篇文章主要為大家介紹了react源碼中的生命周期和事件系統(tǒng)實(shí)例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01

最新評(píng)論