共计 2157 个字符,预计需要花费 6 分钟才能阅读完成。
别离应用 Vue2 和 Vue3 创立一个组件,用一个对象数组作为组件的状态,以它的长度作为变量,考查 Vue2 和 Vue3 性能。
内存占用
数组长度 | Vue2 | Vue3 |
---|---|---|
1 | 10.2 MB | 11.1 MB |
10 000 | 17.9 MB | 12.1 MB |
100 000 | 67.4 MB | 14.4 MB |
1 000 000 | 568 MB | 36.0 MB |
初始化工夫
数组长度 | Vue2 | Vue3 |
---|---|---|
1 | 7.2 ms | 7.8 ms |
10 000 | 110 ms | 6.9 ms |
100 000 | 803 ms | 6.7 ms |
1 000 000 | 2282 ms | 7.0 ms |
测试代码
Vue2
<!DOCTYPE html> | |
<html lang="en"> | |
<head> | |
<meta charset="UTF-8"> | |
<meta http-equiv="X-UA-Compatible" content="IE=edge"> | |
<meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
<title>Document</title> | |
<script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script> | |
</head> | |
<body> | |
<div id="app">{{message}}</div> | |
<script> | |
const len = 1000000 | |
const arr = new Array(len) | |
for (let i = 0; i < len; i++) {arr[i] = {id: i, name: 'test'} | |
} | |
console.time('vue2:') | |
new Vue({ | |
el: '#app', | |
data() { | |
return { | |
message: 'Hello Vue!', | |
arr | |
} | |
}, | |
created() {console.timeEnd('vue2:') | |
} | |
}) | |
</script> | |
</body> | |
</html> |
Vue3
<!DOCTYPE html> | |
<html lang="en"> | |
<head> | |
<meta charset="UTF-8"> | |
<meta http-equiv="X-UA-Compatible" content="IE=edge"> | |
<meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
<title>Document</title> | |
<script src="https://unpkg.com/vue@3/dist/vue.global.js"></script> | |
</head> | |
<body> | |
<div id="app">{{message}}</div> | |
<script> | |
const {createApp} = Vue | |
const len = 1000000 | |
const arr = new Array(len) | |
for (let i = 0; i < len; i++) {arr[i] = {id: i, name: 'test'} | |
} | |
console.time('vue3:') | |
createApp({data() { | |
return { | |
message: 'Hello Vue!', | |
arr | |
} | |
}, | |
created() {console.timeEnd('vue3:') | |
} | |
}).mount('#app') | |
</script> | |
</body> | |
</html> |
揣测
Vue2 应用 Object.defineProperty()
创立响应式属性,初始化时,除了遍历 props
和data
的每个属性,还会深度遍历子对象和子数组,从新定义它们的属性,并创立如 Observer
和Dep
实例,来察看属性变动,并且 Object.defineProperty()
重写的 get
和set
办法也是挂载在实例上。深度遍历加上创立这些实例的花销远大于原始对象,于是,能够看到 Vue2 无论是内存,还是初始化工夫随着数组长度减少猛涨,和 Vue3 产生了显著的差距。
当数组长度达到 100 万,Vue3 内存仅减少 25 MB,但 Vue2 内存暴增了 500+ MB,这些多进去的内存便是创立这些实例产生的额定开销。察看 Vue2 内存快照,发现 Vue2 创立了 100 万个 Observer
实例,相当于每个对象一个(另外三个 Observer
实例别离用于察看组件 props
、data
和数组自身)。数组中每个对象有两个属性,相应创立 200 万个 Dep
实例,100 万个 Observer
实例相应创立了 100 万个 Dep
实例,共创立超过 300 万个 Dep
实例。
初始化工夫来看,Vue3 简直不受数组长度影响,而 Vue2 则随数组长度显著变长。这是因为 Vue3 应用 Proxy
类实现响应式,零碎会代理响应式对象的操作,如组件的 props
和data
,包含属性读取、赋值等,不会进行深度遍历。当读取响应式对象的属性时,才会创立 Dep
实例,如果读取的属性值也是对象时,再将这个对象响应式化,能够说 Vue3 是“懒”响应式。因而,初始化过程中 Vue3 运行工夫简直不变,如果不去操作这些属性,也简直不会有太多内存耗费。
列表渲染
别离应用 Vue2 和 Vue3 渲染 50000 个元素:
<ul> | |
<li v-for="item in arr">{{item.id}}</li> | |
</ul> |
能够看出,无论是内存占用或渲染实现工夫,Vue3 更占优势:
Vue2 | Vue3 | |
---|---|---|
内存 | 124 MB | 53.1 MB |
渲染实现工夫 | 862ms | 284 ms |