手写JS之vue更新数据不会更新在页面上的问题
问题背景
近期开发推荐系统需求的时候需要在后台取新数据,所以需要重新给recommendData对象添加一条数据。自己尝试了一番,发现果真不可以直接赋值。可用语句如下:
Vue.set(this.recommendData,index,result.body[0].object);//使视图及时更新
if(this.recommendData[index]) console.log(this.recommendData[index].length)
这句话的作用是给recommendData对象数组的第index项赋值。
数组更新检测
因为 JavaScript 的限制,Vue.js 不能检测到下面数组变化:
- 当你利用索引直接设置一个项时:
vm.items[indexOfItem] = newValue
- 当你修改数组的长度时,例如:
vm.items.length = newLength
- 当你删除对象中某一个
key
值时,例如:delete vm.items[key]
可以用vue自己封装的Vue.delete( target, key )
对象更改检测注意事项(由于 JavaScript 的限制,Vue 不能检测对象属性的添加或删除:)
var vm = new Vue({
data: {
a:1
}
})
//vm.a ok
vm.b=2
//vm.b no
对于已经创建的实例,Vue 不能动态添加根级别的响应式属性。但是,可以使用 Vue.set(object, key, value)
方法向嵌套对象添加响应式属性。例如,对于:
var vm = new Vue({
data:{
userProfile: {
name:'Anika'
}
}
})
可以添加一个新的age属性到嵌套的userProfile对象
Vue.set(vm.userProfile,'age',27)
//vm.$set(this.userProfile,'age',27)
有时你可能需要为已有对象赋予多个新属性,比如使用 Object.assign()
或 _.extend()
。在这种情况下,你应该用两个对象的属性创建一个新的对象。
Object.assign(this.userProfile,{
age:27,
favoriteColor:'Vue Green'
}) //no
this.userProfile = Object.assign({},this.userProfile,{
age:27,
favoriteColor:'Vue Green'
})
vue说明文档
又看Vue的说明文档上说关于变更方法的问题:
Vue 将被侦听的数组的变更方法进行了包裹,所以它们也将会触发视图更新。这些被包裹过的方法包括:
push()
pop()
shift()
unshift()
splice()
sort()
reverse()
你可以打开控制台,然后对前面例子的 items
数组尝试调用变更方法。比如 example1.items.push({ message: 'Baz' })
。
替换数组
变更方法,顾名思义,会变更调用了这些方法的原始数组。相比之下,也有非变更方法,例如 filter()
、concat()
和 slice()
。它们不会变更原始数组,而总是返回一个新数组。当使用非变更方法时,可以用新数组替换旧数组:
example1.items = example1.items.filter(function (item) {
return item.message.match(/Foo/)
})
你可能认为这将导致 Vue 丢弃现有 DOM 并重新渲染整个列表。幸运的是,事实并非如此。Vue 为了使得 DOM 元素得到最大范围的重用而实现了一些智能的启发式方法,所以用一个含有相同元素的数组去替换原来的数组是非常高效的操作。