代码拆解,vue虚拟滚动长列表的实现

1、需求产生:

        在项目中遇到一个问题。公司推出的移动端设备配置低,导致商家添加过多商品时加载大量dom元素,设备本身运存低的情况下,导致整个移动端页面异常卡顿。

2、解决思路:       

        本身想通过数据分页加载来解决此问题,后发现不行。如果只是一两页数据还好,当数据全部加载完后,又会复现之前的卡顿问题。

        因此换解决方案:采用虚拟滚动。即 通过虚拟滚动(列表),即数据滚动,dom元素固定个数不再新增。达到日常所见列表项所展示效果。

3、具体展开

        

  4、Demo实现:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>虚拟滚动(虚拟列表)</title>
    <style>
        .list-wrap{
            position: relative;
            overflow-y: scroll;
            width: 200px;
            margin: 100px auto;
            box-sizing: border-box;
            border: solid 1px;
        }
        .list{
            position: absolute;
            top: 0;
            left: 0;
            list-style: none;
        }
        .list li{
            border: solid 1px;
            line-height: 30px;
        }
</style>
</head>
<body>
<ul id="app">
    <div class="list-wrap" ref="listWrap" @scroll="scrollListener">
        <!-- 虚拟列表所有项的高度 -->
        <div class="scroll-bar" ref="scrollBar"></div>
        <!-- 虚拟列表项 -->
        <ul class="list" ref="list">
            <li v-for="val in showList">
                {{val}}
            </li>
        </ul>
    </div>
</ul>
<!-- 引Vue -->
<script src="https://cdn.jsdelivr.net/npm/vue@2"></script>
<script>
    new Vue({
        el: '#app',
        data(){
            return {
                list: [],// 超长的显示数据
                itemHeight: 30,// 每一列的高度
                showNum: 10,// 显示几条数据
                start: 0,// 滚动过程显示的开始索引
                end: 10,// 滚动过程显示的结束索引
            }
        },
        computed: {
            // 计算可视窗口内要显示的数组 从长数组中截取要展示的头尾
            showList(){
                return this.list.slice(this.start, this.end);
            }
        },
        mounted(){
            // 列表项
            for (let i = 0; i < 1000000; i++) {
                this.list.push('第' + i + '个')
            }
            // 计算滚动容器高度  每一项高度 * 可视窗口要展示的数据条数
            this.$refs.listWrap.style.height = this.itemHeight * this.showNum + 'px';
            // 计算总的数据需要的高度,构造滚动条高度   每一项高度 * 所有项
            this.$refs.scrollBar.style.height = this.itemHeight * this.list.length + 'px';
        },
        
        methods: {
            scrollListener(){
                // 获取滚动卷去的高度
                let scrollTop = this.$refs.listWrap.scrollTop;
                // 开始的数组索引  卷去高度 / 每一项高度
                this.start = Math.floor(scrollTop / this.itemHeight);
                // 结束索引  开始索引 + 可视窗口内展示的数据条数
                this.end = this.start + this.showNum;
                // 动态计算卷去总高度给 虚拟列表一个顶部的偏移量,让数据一直保持在可视窗口中
                this.$refs.list.style.top = this.start * this.itemHeight + 'px';
            }
        }
    })
</script>
</body>
</html>

5、展示效果:

        以上是 定高虚拟列表的实现(每一项的高度固定)。

         针对每一项高度不一致时,可以采用 不定高虚拟滚动(虚拟列表)icon-default.png?t=M666http://t.csdn.cn/cLpuM        

如有不对或表达有误的地方请大佬及时指出,小弟定及时改正。


版权声明:本文为m0_64246444原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。