牛骨文教育服务平台(让学习变的简单)
博文笔记

vue列表页渲染优化

创建时间:2017-07-24 投稿人: 浏览次数:510

想法

初始化时,vue会对data做getter、setter改造,在现代浏览器里,虽然JS已经足够快,但仍然有优化空间。

列表页的数据结构为:

list: [
        // 每一项有不同的来源,不同来源的数据都不同,因此放到一个数组里
        [{
            sourceId: "xmla", // 来源的唯一标识
            id: 3001, // 资源的唯一标识
            source: "喜马拉雅", // 来源
            title: "昆曲之牡丹亭",
            imageUrl: "http://x.baidu.com/x.gif",
            album: 0,  // 是否是专辑
            hot: 1345,
            anchor: "青雪"
        }, {
            sourceId: "xmla", // 来源的唯一标识
            id: 3005, // 资源的唯一标识
            source: "手机百度", // 来源
            title: "昆曲之牡丹亭",
            imageUrl: "http://x.baidu.com/x.gif",
            album: 0,  // 是否是专辑
            hot: 1345,
            anchor: "青雪"
        }],
        [{
            sourceId: "xmla",
            id: 3002, // 资源的唯一标识
            source: "喜马拉雅", // 来源
            title: "昆曲之春江花月夜",
            imageUrl: "http://x.baidu.com/x.gif",
            album: 0,  // 是否是专辑
            hot: 1345,
            anchor: "青雪"
        }],
    ]
}

Vue会给数组中的每个值设置getter和setter来监听它们的变动
但其实列表数据是不会发生变化的,这些操作是多余的。

方法一:使用Object.freeze()

Object.freeze()是ES5新增的API,用来冻结一个对象,禁止对象被修改。vue 1.0.18+以后,不会对已冻结的data做getter、setter转换。

如果确保某个data不需要跟踪依赖,可以使用Object.freeze将其冻结。需要注意的是,被冻结的是对象的值,仍然可以将引用整个替换调。看下面例子:

<p v-for="item in list">{{ item.value }}</p>
new Vue({
    data: {
        // vue不会对list里的object做getter、setter绑定
        list: Object.freeze([
            { value: 1 },
            { value: 2 }
        ])
    },
    created () {
        // 界面不会有响应
        this.list[0].value = 100;

        // 下面两种做法,界面都会响应
        this.list = [
            { value: 100 },
            { value: 200 }
        ];
        this.list = Object.freeze([
            { value: 100 },
            { value: 200 }
        ]);
    }
})

当使用Vuex进行状态管理时,应当在给state.xxx赋值前使用Object.freeze():

[LIST_INIT](state, {list}) {
    Object.freeze(list);
    state.list = list;
},

getter和setter没有了。

方法二:传string方法

由于从后端取回的数据本身为字符串,不进行JSON.parse()直接存在state中,即可阻止Vue的改造。
使用时,在页面组件中引入字符串,JSON.parse()后可以直接赋值给this.XXX,如有需要还可以进一步子组件。

this.test = {
    a:{
        c:1,
        d:2
    },
    b:2
}
<list-item :test="test.a"></list-item>
声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。