后端一次性返回十萬條數據時,前端需要采用多種性能優化策略來避免頁面卡頓

當后端一次性返回十萬條數據時,前端需要采用多種性能優化策略來避免頁面卡頓。以下是主要的優化方案:

  1. 分頁加載 - 將數據分批次加載顯示
  2. 虛擬滾動 - 只渲染可視區域內的數據
  3. 數據懶加載 - 按需加載數據
  4. Web Workers - 在后臺線程處理數據
  5. 時間切片 - 分散渲染任務避免阻塞主線程

下面是具體的實現代碼:

<template><div class="large-data-view"><h2>大數據量處理示例</h2><!-- 性能監控 --><div class="performance-info"><span>總數據量: {{ totalDataCount }}</span><span>當前顯示: {{ startIndex + 1 }} - {{ Math.min(startIndex + pageSize, totalDataCount) }}</span><span>渲染耗時: {{ renderTime }}ms</span></div><!-- 分頁控件 --><div class="pagination-controls"><button @click="prevPage" :disabled="currentPage <= 1">上一頁</button><span>第 {{ currentPage }} 頁,共 {{ totalPages }} 頁</span><button @click="nextPage" :disabled="currentPage >= totalPages">下一頁</button><select v-model="pageSize" @change="onPageSizeChange"><option value="50">每頁50條</option><option value="100">每頁100條</option><option value="200">每頁200條</option><option value="500">每頁500條</option></select></div><!-- 虛擬滾動列表 --><div class="virtual-scroll-container" ref="scrollContainer" @scroll="onScroll"><div class="scroll-placeholder" :style="{ height: totalHeight + 'px' }"></div><div class="visible-items" :style="{ transform: `translateY(${offsetY}px)` }"><div class="data-item" v-for="item in visibleItems" :key="item.id":style="{ height: itemHeight + 'px' }"><span class="item-index">{{ item.id }}</span><span class="item-name">{{ item.name }}</span><span class="item-value">{{ item.value }}</span><span class="item-time">{{ formatTime(item.timestamp) }}</span></div></div></div><!-- 加載指示器 --><div v-if="loading" class="loading-indicator">數據加載中...</div></div>
</template><script setup lang="ts">
import { ref, computed, onMounted, nextTick } from 'vue';// 響應式數據
const allData = ref<any[]>([]);
const startIndex = ref(0);
const pageSize = ref(100);
const itemHeight = 50; // 每項高度
const loading = ref(false);
const renderTime = ref(0);
const scrollContainer = ref<HTMLElement | null>(null);// 分頁相關數據
const currentPage = computed(() => Math.floor(startIndex.value / pageSize.value) + 1);
const totalPages = computed(() => Math.ceil(allData.value.length / pageSize.value));
const totalDataCount = computed(() => allData.value.length);// 虛擬滾動相關數據
const visibleCount = ref(15); // 可視區域項目數
const scrollTop = ref(0);
const offsetY = ref(0);
const totalHeight = computed(() => allData.value.length * itemHeight);// 可視區域的數據
const visibleItems = computed(() => {const start = Math.max(0, Math.floor(scrollTop.value / itemHeight));const end = Math.min(allData.value.length, start + visibleCount.value);return allData.value.slice(start, end);
});// 方法
const formatTime = (timestamp: number) => {const date = new Date(timestamp);return date.toLocaleTimeString();
};// 分頁方法
const prevPage = () => {if (currentPage.value > 1) {startIndex.value = (currentPage.value - 2) * pageSize.value;scrollToTop();}
};const nextPage = () => {if (currentPage.value < totalPages.value) {startIndex.value = currentPage.value * pageSize.value;scrollToTop();}
};const onPageSizeChange = () => {startIndex.value = 0;scrollToTop();
};const scrollToTop = () => {if (scrollContainer.value) {scrollContainer.value.scrollTop = 0;}
};// 虛擬滾動處理
const onScroll = () => {if (scrollContainer.value) {scrollTop.value = scrollContainer.value.scrollTop;offsetY.value = Math.floor(scrollTop.value / itemHeight) * itemHeight;}
};// 模擬加載大數據
const loadData = async () => {loading.value = true;const startTime = performance.now();try {// 模擬從后端獲取10萬條數據// 實際項目中這里應該是API調用const data = generateLargeData(100000);allData.value = data;// 使用時間切片技術處理數據渲染await nextTick();const endTime = performance.now();renderTime.value = Math.round(endTime - startTime);} finally {loading.value = false;}
};// 生成模擬數據
const generateLargeData = (count: number) => {const data = [];for (let i = 0; i < count; i++) {data.push({id: i + 1,name: `數據項 ${i + 1}`,value: Math.random() * 1000,timestamp: Date.now() - Math.floor(Math.random() * 10000000)});}return data;
};// 初始化
onMounted(() => {loadData();// 計算可視區域能顯示的項目數if (scrollContainer.value) {visibleCount.value = Math.ceil(scrollContainer.value.clientHeight / itemHeight) + 5;}
});
</script><style scoped>
.large-data-view {height: 100%;display: flex;flex-direction: column;padding: 20px;
}.performance-info {display: flex;gap: 20px;margin-bottom: 20px;padding: 10px;background-color: #f5f5f5;border-radius: 4px;
}.pagination-controls {display: flex;justify-content: space-between;align-items: center;margin-bottom: 20px;padding: 10px;background-color: #f9f9f9;border-radius: 4px;
}.pagination-controls button {padding: 6px 12px;background-color: #007bff;color: white;border: none;border-radius: 4px;cursor: pointer;
}.pagination-controls button:disabled {background-color: #ccc;cursor: not-allowed;
}.pagination-controls select {padding: 6px;border: 1px solid #ddd;border-radius: 4px;
}.virtual-scroll-container {flex: 1;border: 1px solid #ddd;border-radius: 4px;overflow: auto;position: relative;
}.scroll-placeholder {position: absolute;left: 0;top: 0;right: 0;z-index: -1;
}.visible-items {position: absolute;left: 0;right: 0;top: 0;z-index: 1;
}.data-item {display: flex;align-items: center;padding: 0 15px;border-bottom: 1px solid #eee;
}.item-index {width: 80px;font-weight: bold;
}.item-name {flex: 1;
}.item-value {width: 120px;text-align: right;
}.item-time {width: 100px;text-align: right;color: #999;font-size: 0.9em;
}.loading-indicator {position: fixed;top: 50%;left: 50%;transform: translate(-50%, -50%);padding: 20px;background-color: rgba(0, 0, 0, 0.8);color: white;border-radius: 4px;z-index: 1000;
}
</style>
/*** 大數據處理工具類* 使用Web Workers和時間切片技術處理大量數據*/// 時間切片處理大數據
export class DataProcessor {private chunkSize: number;constructor(chunkSize: number = 1000) {this.chunkSize = chunkSize;}/*** 分塊處理大數據數組* @param data 大數據數組* @param processor 處理函數* @returns Promise*/async processInChunks<T, R>(data: T[], processor: (item: T) => R): Promise<R[]> {const results: R[] = [];const totalChunks = Math.ceil(data.length / this.chunkSize);for (let i = 0; i < totalChunks; i++) {const start = i * this.chunkSize;const end = Math.min(start + this.chunkSize, data.length);const chunk = data.slice(start, end);// 處理當前塊const chunkResults = chunk.map(processor);results.push(...chunkResults);// 讓出控制權,避免阻塞UIif (i < totalChunks - 1) {await this.yieldToMain();}}return results;}/*** 讓出控制權給主線程*/private yieldToMain(): Promise<void> {return new Promise(resolve => {setTimeout(resolve, 0);});}/*** 使用Web Worker處理數據* @param data 數據* @param workerFunction 處理函數字符串* @returns Promise*/processWithWorker<T, R>(data: T[], workerFunction: string): Promise<R[]> {return new Promise((resolve, reject) => {// 創建Web Workerconst workerCode = `self.onmessage = function(e) {const { data, processor } = e.data;const func = new Function('return ' + processor)();const results = data.map(func);self.postMessage(results);};`;const blob = new Blob([workerCode], { type: 'application/javascript' });const worker = new Worker(URL.createObjectURL(blob));worker.onmessage = function(e) {resolve(e.data);worker.terminate();};worker.onerror = function(error) {reject(error);worker.terminate();};worker.postMessage({data,processor: workerFunction});});}
}// 創建數據處理器實例
export const dataProcessor = new DataProcessor(1000);
<template><div class="infinite-scroll-list" ref="container"><div class="list-container"><div class="list-item" v-for="item in displayItems" :key="item.id"><slot :item="item"></slot></div><div v-if="loading" class="loading-more">加載中...</div><div v-if="noMore" class="no-more">沒有更多數據了</div></div></div>
</template><script setup lang="ts">
import { ref, computed, onMounted, onBeforeUnmount } from 'vue';const props = defineProps<{items: any[];pageSize?: number;threshold?: number; // 距離底部多少像素時觸發加載
}>();const emit = defineEmits<{(e: 'loadMore'): void;
}>();// 默認值
const pageSize = props.pageSize || 50;
const threshold = props.threshold || 100;// 響應式數據
const container = ref<HTMLElement | null>(null);
const displayedCount = ref(pageSize);
const loading = ref(false);
const noMore = ref(false);// 計算屬性
const displayItems = computed(() => {return props.items.slice(0, displayedCount.value);
});// 方法
const handleScroll = () => {if (!container.value) return;const { scrollTop, scrollHeight, clientHeight } = container.value;const distanceToBottom = scrollHeight - scrollTop - clientHeight;// 當距離底部小于閾值且還有數據時觸發加載if (distanceToBottom < threshold && !loading && !noMore.value) {loadMore();}
};const loadMore = () => {if (displayedCount.value >= props.items.length) {noMore.value = true;return;}loading.value = true;// 模擬異步加載setTimeout(() => {displayedCount.value = Math.min(displayedCount.value + pageSize,props.items.length);loading.value = false;if (displayedCount.value >= props.items.length) {noMore.value = true;}}, 300);
};// 暴露方法給父組件
defineExpose({reset() {displayedCount.value = pageSize;noMore.value = false;},setLoading(status: boolean) {loading.value = status;}
});// 生命周期
onMounted(() => {if (container.value) {container.value.addEventListener('scroll', handleScroll);}
});onBeforeUnmount(() => {if (container.value) {container.value.removeEventListener('scroll', handleScroll);}
});
</script><style scoped>
.infinite-scroll-list {height: 100%;overflow-y: auto;
}.list-item {padding: 10px;border-bottom: 1px solid #eee;
}.loading-more,
.no-more {padding: 15px;text-align: center;color: #999;
}
</style>

這些優化方案可以有效解決前端處理大量數據時的卡頓問題:

  1. 分頁加載:將10萬條數據分頁顯示,每次只渲染少量數據
  2. 虛擬滾動:只渲染可視區域內的數據項,大幅減少DOM節點數量
  3. 時間切片:將大數據處理任務分解成小塊,避免長時間阻塞主線程
  4. 按需渲染:根據用戶滾動位置動態加載和卸載數據
  5. 性能監控:實時顯示渲染性能指標,便于調優

通過這些技術的組合使用,即使面對10萬條數據,頁面也能保持流暢的用戶體驗。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/97895.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/97895.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/97895.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

基于-輕量級文檔搜索系統的測試報告

文章目錄一、項目背景二、項目功能三、測試計劃&#xff08;一&#xff09;測試用例設計&#xff08;二&#xff09;測試用例實現1.功能測試2.界面測試3.兼容性測試4.易用性測試5.安全性測試一、項目背景 1.基于輕量級文檔檢索系統采用C技術棧來實現&#xff0c;同時使用了本地…

編輯器vim(Linux)

Linux下開發工具是獨立的寫代碼——編輯器 vim編譯代碼——gcc/g調試——gdb、cgdb構建工具——makefile、make、cmakevim只用來寫代碼注意&#xff1a;直接用vim打開一個不存在的文件并保存退出&#xff0c;就會自動生成該文件vim有多種模式命令模式&#xff08;Normal Mode&a…

GitLab,2025最新如何配置中的SSH key步驟

電腦右鍵先檢查&#xff0c;是否有公鑰 git cat ~/.ssh/id_rsa.pub下面是有&#xff0c;不用生成公鑰&#xff0c;沒有就要生成生成本地電腦公鑰, 建議用第二種 //第一種ssh-keygen -t rsa//第二種------- 1.打開git bash,輸入&#xff1a;ssh-keygen -t rsa -C “你的郵箱”ss…

華為HCIE證書多久續一次費?費用多少?

根據華為官方政策&#xff0c;華為認證HCIE的有效期為3年&#xff0c;有效期自證書正式發放之日起計算&#xff0c;考生可通過華為人才在線官網登錄個人賬號&#xff0c;在“我的證書”欄目中查詢具體有效期起止時間。一、HCIE證書到期后的續證方式 1.重考對應HCIE的認證考試&a…

提升文本到圖像強化學習穩定性:Pref - GRPO算法如何革新圖像生成?

提升文本到圖像強化學習穩定性&#xff1a;Pref - GRPO算法如何革新圖像生成&#xff1f; 在文本到圖像生成領域&#xff0c;強化學習正重塑著模型與人類偏好的對齊方式。本文聚焦于一種創新的基于成對偏好獎勵的GRPO方法&#xff08;Pref - GRPO&#xff09;&#xff0c;它通…

Linux UDisks守護進程曝本地提權漏洞CVE-2025-8067,PoC已發布

漏洞概述安全研究人員在Linux環境中廣泛使用的磁盤管理組件UDisks守護進程中&#xff0c;發現了一個嚴重漏洞&#xff08;編號CVE-2025-8067&#xff0c;CVSS評分8.5&#xff09;。該漏洞已報告給紅帽產品安全團隊&#xff0c;并在UDisks更新版本中得到修復。技術細節該漏洞存在…

uniapp 開發上架 iOS App全流程

操作文檔網址&#xff1a;https://ask.dcloud.net.cn/article/152 操作學習視頻地址&#xff1a;uniapp打包上線微信小程序、安卓、IOS流程_嗶哩嗶哩_bilibili 第一步&#xff1a;注冊蘋果 iOS 個人開發者賬號 費用說明 ?個人開發者賬號?&#xff1a;適用于獨立開發者或小…

Sqlsugar補充自定義模板

DBFirst默認創建所有實體CreateClassFile()的第二個參數為生成實體類命名空間//.net6以下 db.DbFirst.IsCreateAttribute().CreateClassFile("c:\\Demo\\1", "Models"); //.net6以上 string加? db.DbFirst.IsCreateAttribute().StringNullable().CreateCl…

LeetCode 392.判斷子序列

給定字符串 s 和 t &#xff0c;判斷 s 是否為 t 的子序列。 字符串的一個子序列是原始字符串刪除一些&#xff08;也可以不刪除&#xff09;字符而不改變剩余字符相對位置形成的新字符串。&#xff08;例如&#xff0c;"ace"是"abcde"的一個子序列&#x…

邏輯回歸:從原理到實戰的完整指南

在機器學習中&#xff0c;分類任務是最常見的應用場景之一。而邏輯回歸&#xff08;Logistic Regression&#xff09;&#xff0c;盡管名字中有“回歸”&#xff0c;實際上是一種非常強大且廣泛應用的二分類模型。它簡單、高效、可解釋性強&#xff0c;是數據科學初學者入門分類…

鴻蒙搭配前端開發:應用端與WEB端交互

鴻蒙系統&#xff08;HarmonyOS&#xff09;是華為開發的一款面向全場景的分布式操作系統&#xff0c;其設計初衷是為了適應物聯網時代的需求&#xff0c;旨在構建一個統一的操作系統&#xff0c;支持多種設備的無縫協同工作。其分布式開發的一些主要優勢&#xff1a; 跨設備協…

配置sscms時被sql server處處刁難

今天要記下來的是一個小例子。接前面&#xff0c;當我終于完成sql server的安裝時&#xff0c;才發現要填寫sscms的兩個空是有多么艱難。首先安裝sql server2016出現了太多環境不兼容的問題&#xff0c;讓我只好退而安裝sql server2012。安裝sql server2012時其實是可以避坑的&…

【Flink】DataStream API:源算子、數據類型

目錄源算子&#xff08;Source&#xff09;從集合中讀取數據從文件讀取數據從Socket讀取數據從Kafka讀取數據從數據生成器讀取數據Flink支持的數據類型Flink的類型系統Flink支持的數據類型類型提示&#xff08;Type Hints&#xff09;源算子&#xff08;Source&#xff09; Fli…

Linux 安裝docker-compose安裝方法(安裝docker compose安裝)

文章目錄**方法一&#xff1a;通過 curl 下載二進制文件&#xff08;推薦&#xff09;**1. 安裝前準備- **確保已安裝 Docker**- **檢查 Docker 是否安裝成功**2. 下載并安裝 Docker Compose- **下載最新版本的 Docker Compose 二進制文件**- **國內加速下載&#xff08;解決 G…

OCR 發票識別與驗真接口:助力電子化發票新時代

自 2025 年 10 月 1 日起&#xff0c;紙質火車票徹底告別歷史舞臺&#xff0c;全面數字化的電子發票取而代之&#xff0c;這一變革標志著票務領域的重大革新&#xff0c;也讓電子化發票處理的需求呈井噴式增長。在此背景下&#xff0c;OCR 發票識別和發票驗真接口技術挺身而出&…

設計模式:抽象工廠模式(Abstract Factory Pattern)

文章目錄一、概念二、實例分析三、完整示例一、概念 抽象工廠模式是一種創建型設計模式。 提供一個接口用于創建一系列相關或相互依賴的對象&#xff0c;而無需指定它們的具體類。 相比于工廠方法模式&#xff0c;抽象工廠模式不僅僅是創建單一產品&#xff0c;而是一族產品&am…

輕量級注意力模型HOTSPOT-YOLO:無人機光伏熱異常檢測新SOTA,mAP高達90.8%

【導讀】 無人機光伏巡檢如何更智能、更高效&#xff1f;HOTSPOT-YOLO模型給出了亮眼答案&#xff01;給AI裝上“熱成像鷹眼”&#xff0c;能精準鎖定光伏板上的細微熱斑缺陷。它不僅將檢測精度&#xff08;mAP&#xff09;提升至90.8%&#xff0c;更在保持實時性的前提下大幅…

CHT共軛傳熱: 導熱系數差異如何影響矩陣系數

文章目錄 一、導熱系數差異如何影響矩陣系數&#xff1f;二、如何處理系數差異以加速收斂&#xff1f;1. **變量重縮放&#xff08;Scaling of Variables&#xff09;**2. **使用物理型預條件子&#xff08;Physics-based Preconditioning&#xff09;**3. **區域分解法&#x…

Vue Vapor 事件機制深潛:從設計動機到源碼解析

基于 vue3.6&#xff08;alpha 階段&#xff09;及 Vapor 的最新進展撰寫&#xff1b;Vapor 仍在演進中&#xff0c;部分實現可能繼續優化。TL;DR&#xff08;速覽&#xff09; 傳統&#xff08;≤3.5&#xff09;&#xff1a;事件以元素為中心綁定&#xff1b;每個元素用 el._…

Day 01(01): Hadoop與大數據基石

目標&#xff1a;建立對大數據生態的整體認知&#xff0c;理解HDFS和MapReduce的核心思想。 8:00-9:30&#xff1a;【視頻學習】在B站搜索“Hadoop入門”或“三小時入門大數據”&#xff0c;觀看1-2個高播放量的簡介視頻&#xff0c;了解大數據面臨的問題和Hadoop的解決方案。 …