美文网首页
element el-table 处理万数级别表格不卡顿秒开

element el-table 处理万数级别表格不卡顿秒开

作者: Rogi | 来源:发表于2020-08-12 23:36 被阅读0次

    先看效果图吧,先上100条内容,介绍思路最后上2万条


    效果图

    实现思路,监听滚动事件,获取滚动条到顶部距离通过除以单条数的DOM高度,以这个数作为显示的第一条数据的index,然后以刚刚获取到的index + 屏幕高度 / 单条数的DOM高度得到一屏能放下多少条数据,这样子就能准确获取一屏要显示的数据
    这种做法只需要vue渲染10~20条数据,而不用几万条几千条生成,加上内嵌的DOMDOM数量直接爆炸
    所以只需要渲染这十几条数据就能达到巨额的DOM优化,如果看不懂没关系,以下有一幅图可以帮助你理解

    原理图

    这里有个demo可以复制看一下效果

    <html lang="zh">
    <head>
        <meta charset="UTF-8">
        <meta name="viewport"
              content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
        <meta http-equiv="X-UA-Compatible" content="ie=edge">
        <title>Document</title>
        <style>
            .container {
                width: 300px;
                height: 600px;
                overflow: auto;
                border: 1px solid;
                margin: 100px auto;
            }
            .item {
                height: 29px;
                line-height: 30px;
                border-bottom: 1px solid #aaa;
                padding-left: 20px;
            }
        </style>
    </head>
    <body>
    <div id="app">
        <button @click="add">增加</button>
        <div class="container" ref="container">
            <div class="scroll-wrapper" :style="style">
                <div v-for="(item, index) in scrollList" :key="index" class="item">{{item}}</div>
            </div>
        </div>
    </div>
    <script src="https://cdn.jsdelivr.net/npm/vue"></script>
    <script>
        new Vue({
            el: '#app',
            data: {
                list: [
                    '测试数据'
                ],
                startIndex: 0,
                endIndex: 60,
                paddingTop: 0,
                paddingBottom: 0,
                allHeight: 0
            },
            computed: {
                scrollList() {
                    return this.list.slice(this.startIndex, this.endIndex)
                },
                style() {
                    return {
                        paddingTop: this.paddingTop + 'px',
                        paddingBottom: this.paddingBottom + 'px'
                    }
                }
            },
            watch: {
                list(val) {
                    const valLen = val.length
                    this.allHeight = valLen * 30
                    this.paddingBottom = this.allHeight - this.scrollList.length * 30
                }
            },
            mounted() {
                const container = this.$refs.container
                container.addEventListener('scroll', () => {
                    const top = container.scrollTop
                    this.startIndex = Math.floor(top / 30)
                    this.endIndex = this.startIndex + 21
     
                    this.paddingTop = top
                    if (this.endIndex >= this.list.length - 1) {
                        this.paddingBottom = 0
                        return
                    }
                    console.log(this.paddingBottom)
                    this.paddingBottom = this.allHeight - 600 - top
                })
            },
            methods: {
                add() {
                    let arr = new Array(50000).fill(0)
                    arr = arr.map((item, index) => {
                        return index
                    })
                    this.list = [
                        ...this.list,
                        ...arr
                    ]
                }
            }
        })
    </script>
    </body>
    </html>
    

    解决完dom过大问题之后,还有一个问题,如果遇到dom复杂例如element的组件也要处理这类这么大数据量的表单显然是会卡顿的。
    解决问题之前我们先思考一下用以上的方法为什么会造成卡顿呢?
    问题其实出在高频触发DOM的更新,由于vue是使用双向绑定进行页面更新的,每次绑定的数据更新都会触发页面的DOM进行更新,所以导致了,我们鼠标每滚动一次都会重新渲染和更新这十几条信息,无疑也是高额的性能消耗。
    解决方法其实也很简单,我们要利用vue在key相同的情况下是会缓存DOM的特性,让DOMkey相同,但是渲染的数据却不同,就能缓存元素了

    最后是2万条数据在element实现加载2万条数据



    这是本次渲染的速度


    相关文章

      网友评论

          本文标题:element el-table 处理万数级别表格不卡顿秒开

          本文链接:https://www.haomeiwen.com/subject/rdlqphtx.html