虚拟列表 如何 计算页面能 显示多少列表项 ?

大家在开发网页时是否遇到过这样的问题:一个超长列表在页面上渲染时,浏览器突然变得非常卡顿?这是因为浏览器一次性渲染了太多元素导致的。要解决这个问题,我们需要先弄清楚:到底有多少列表项是用户真正能看到的?

为什么要计算最大容纳数量?

想象一下,你有一个包含1000个项目的列表,但用户的屏幕只能同时显示10个。如果一次性全部渲染,浏览器需要创建1000个DOM元素,这会消耗大量内存和计算资源,导致页面滚动卡顿、响应缓慢。

聪明的做法是:只渲染用户能看到的那部分内容!当用户滚动时,再动态替换显示的内容。这种方式叫做"虚拟滚动",而要实现它,第一步就是计算可视区域内最多能容纳多少个列表项。

如何计算可视区域的最大容量?

计算原理其实很简单:

  1. 获取容器高度:首先要知道列表容器有多高
  2. 获取单个项目高度:每个列表项有多高
  3. 简单除法计算:容器高度 ÷ 单个项目高度 = 基本可见项目数

但实际应用中还需要考虑一个重要情况:当用户滚动时,顶部和底部可能会各出现一个"未完全显示"的项目。所以最终公式是:

js 复制代码
可见项目数 = Math.ceil(容器高度 / 项目高度) + 1

这里的Math.ceil()是向上取整,确保即使有部分显示的项目也被计算在内。

代码实现解析

javascript 复制代码
// 使用useCallback和throttle优化性能,防止频繁计算
const changeHeight = useCallback(throttle(() => {
  // 获取容器当前的实际高度
  curContainerHeight.current = containerRef.current.offsetHeight
  // 计算最大可见项目数
  curViewNum.current = Math.ceil(curContainerHeight.current / itemHeight) + 1
}, 500), [])

这段代码做了两件事:

  1. 获取列表容器的高度(因为容器高度可能会变化)
  2. 计算并保存当前可视区域内最多能显示多少列表项

处理浏览器窗口大小变化

用户可能会调整浏览器窗口大小,这会改变可视区域的高度,所以我们需要监听resize事件:

javascript 复制代码
useEffect(() => {
  changeHeight() // 初始计算
  window.addEventListener('resize', changeHeight) // 监听窗口变化
  
  return () => {
    window.removeEventListener('resize', changeHeight) // 清理监听器
  }
}, [changeHeight])

这里使用了throttle函数(节流),确保即使窗口被连续调整大小,计算操作最多每500毫秒执行一次,避免过于频繁的计算影响性能。

实际应用价值

通过这种计算,我们可以实现:

  • 只渲染用户能看到的内容,极大提升页面性能
  • 动态适应不同屏幕尺寸和设备方向变化
  • 提供流畅的滚动体验,即使面对海量数据

这种技术在现代Web应用中被广泛使用,如社交媒体feed流、大型数据表格、聊天消息记录等场景。

总结

计算可视区域最大容纳数量是实现高效列表渲染的关键第一步。通过简单的数学计算和适当的事件监听,我们可以显著提升页面性能,为用户提供更流畅的体验。

相关推荐
RoyLin12 小时前
C++ 基础与核心概念
前端·后端·node.js
记得坚持12 小时前
vue2插槽
前端·vue.js
臣臣臣臣臣什么臣12 小时前
uni-app 多文件上传:直接循环调用 uni.uploadFile 实现(并行 / 串行双模式)
android·前端
带只拖鞋去流浪12 小时前
Vue.js响应式API
前端·javascript·vue.js
Coder_R12 小时前
如何 把 Mac 上的 APK(app) 安装到安卓手机上?
前端·面试
前端小灰狼12 小时前
Ant Design Vue Vue3 table 表头筛选重置不清空Bug
前端·javascript·vue.js·bug
前端付豪12 小时前
11、JavaScript 语法:到底要不要写分号?一文吃透 ASI 与坑点清单
前端·javascript
Copper peas12 小时前
Vue 中的 v-model 指令详解
前端·javascript·vue.js
前端小书生12 小时前
NestJs
前端·nestjs
万少12 小时前
十行代码 带你极速接入鸿蒙6新特性 - 应用内打分评价
前端·harmonyos