关于vuex更新视图引发的一些思考详析

目录
  • 场景
  • 解决
  • 总结
  • 扩展

vuex可以集中式存储管理应用的所有组件的状态,当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新

但是,最近踩了vuex的坑:

场景

  • 第一次进入页面加载数据,数据不显示,点击某个按钮或者切换页面后,数据会展示出来
  • 通过分析发现,当第一次加载页面的时候,获取数据的数据为{} (空对象),当数据获取完毕,执行commit()
  • 而此时通过commit()已经改变了state中的数据,在页面中通过computed也可以获取更新后的数据。但是视图没有更新,获取的数据没有展示出来

解决

  • 尝试在页面中发送commit()来再次更新视图,无效
  • 将Vue.set(state, 'myData', data),无效
  • ......

最终方案:

代码中所声明的对象里面的key是动态的,所以初始只声明了{ }。

原来,数据是数组的时候,不能通过索引直接进行赋值,也不能修改数组的长度。 而Vuex只会跟踪在对象创建时就存在的属性,新添加到对象上的新属性不会触发更新。

并且循环嵌套层级太深,视图也可能不更新

最终,给数组对象赋值,这里转化了一下写法,生效。具体如下:

const store = new Vuex.Store({
  state: {
    myData: []
  },
  mutations: {
    setData(state, data) {
      // state.myData = data  不更新视图
      Vue.set(state, 'myData', JSON.parse(JSON.stringify(data)))
    }
  },
  actions: {
    myFn () {
      store.commit('setData')
    }
  }
})

export default store

如果添加了新的对象属性,也可以使用Object.assign(),但必须深拷贝(否则加到对象上的新属性不会触发更新)

let someObject = Object.assign({}, someObject, {newField: value}) // 深拷贝
// Object.assign(this.someObject, { key: 1 }) // 浅拷贝

总结

  • 不能直接去改变 store 中的状态。改变 store 中的状态的需要提交 (commit) mutation
  • 在组件中调用 store 中的状态在计算属性中返回即可获取,也可以直接$store.state来获取
computed: {
    myData () {
      return store.state.myData
    }
},
mounted() {
    console.log(this.store.state.myData)
}
  • 为数组添加新属性并不会触发视图更新,因为vue没有给新属性增加get和set监听
  • 赋值的数据,如果循环嵌套层级太深,可能会导致视图不更新

扩展

这些方法操作数组,vue可以检测到数据变化:

push()
pop()
shift()
unshift()
splice()
sort()
reverse()

而下面这些不会改变原数组(返回了新数组),vue检测不到:

filter()
concat()
slice()

到此这篇关于vuex更新视图引发的一些思考的文章就介绍到这了,更多相关vuex更新视图内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • vue视图不更新情况详解

    我们可能经常会在处理vue项目的时候,遇到数据变化,但是视图并没有实时渲染的情况 vue视图为什么不渲染页面的原因 当你把一个普通的 JavaScript 对象传给 Vue 实例的 data 选项,Vue 将遍历此对象所有的属性,并使用 Object.defineProperty 把这些属性全部转为 getter/setter.Object.defineProperty 是 ES5 中一个无法 shim 的特性,这也就是为什么 Vue 不支持 IE8 以及更低版本浏览器 划重点!!!! 这些 g

  • vue.js移动数组位置,同时更新视图的方法

    使用vue.js v-for绑定若干个选项,需要对选项进行排序上下移动操作. 需要对options里面数组的位置进行交换,通常是这样来写: 假设向前移动一个: var index = this.options.indexOf(option); //获取当前选项对象在数组里面的索引. var tempOption = this.options[index-1]; //存储前一个 this.options[index-1] = option;(this.options[index]) this.op

  • 关于vuex更新视图引发的一些思考详析

    目录 场景 解决 总结 扩展 vuex可以集中式存储管理应用的所有组件的状态,当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新 但是,最近踩了vuex的坑: 场景 第一次进入页面加载数据,数据不显示,点击某个按钮或者切换页面后,数据会展示出来 通过分析发现,当第一次加载页面的时候,获取数据的数据为{} (空对象),当数据获取完毕,执行commit() 而此时通过commit()已经改变了state中的数据,在页面中通过com

  • vue3结构赋值失去响应式引发的问题思考

    目录 前言 原始值的响应式系统的实现 为什么ES6 解构,不能随意使用会破坏他的响应式特性 proxy背景 实现原理 解构 props 对象,因为它会失去响应式 直接赋值reactive响应式对象 vuex中组合API赋值 结语 前言 vue3发布以来经历两年风头正盛,现在大有和react 平分天下的势头,我们知道他是基于proxy 实现响应式的能力, 解决了vue2所遗留下来的一些问题,同时也正由于proxy的特性,也提高了运行时的性能 凡事有利有弊, proxy虽然无敌,但是他也有本身的局限

  • 浅谈Python由__dict__和dir()引发的一些思考

    关于__dict__和dir()的区别和作用请参考这篇文章: 基于Python __dict__与dir()的区别详解 说下我当时遇到的问题: class Demo: def __init__(self, name, age): self.name = name self.age = age def func(self): print('Hello {0}'.format(self.name)) >>> d1 = Demo('Pythoner', 24) >>> has

  • 针对distinct疑问引发的一系列思考

    有人提出了这样一个问题,整理出来给大家也参考一下 假设有如下这样一张表格: 这里的数据,具有如下的特征:在一个DepartmentId中,可能会有多个Name,反之也是一样.就是说Name和DepartmentId是多对多的关系. 现在想实现这样一个查询:按照DepartmentID排完序之后(第一步),再获取Name列的不重复值(第二步),而且要保留在第一步后的相对顺序.以本例而言,应该返回三个值依次是:ACB 我们首先会想到下面这样一个写法 select distinct name from

  • 关于AngularJS中ng-repeat不更新视图的解决方法

    最近写AngularJS项目中,遇到一个问题,先对数组进行赋值,ng--repeat正常工作,然后对数组进行修改,ng-repeat似乎没有工作,视图没有更新. 原因是ng-repeat会已默认值排序,由于我的数组中的元素有重复,所以ng-repeat没有对重复的元素进行刷新,只要对ng-repeat指定track by就可以了 <tr ng-repeat="selectedCriterias in selectedCriteriasArray track by $index"&

  • mysql视图之创建可更新视图的方法详解

    本文实例讲述了mysql视图之创建可更新视图的方法.分享给大家供大家参考,具体如下: 我们知道,在mysql中,视图不仅是可查询的,而且是可更新的.这意味着我们可以使用insert或update语句通过可更新视图插入或更新基表的行. 另外,我们还可以使用delete语句通过视图删除底层表的行.但是,要创建可更新视图,定义视图的select语句不能包含以下任何元素: 聚合函数,如:min,max,sum,avg,count等. DISTINCT子句 GROUP BY子句 HAVING子句 左连接或

  • PostgreSQL 更新视图脚本的注意事项说明

    项目最早是基于Oracle的,移植到PostgreSQL后,本着尽量少修改的原则,创建/更新视图的脚本也沿用了Oracle风格的CREATE OR REPLACE VIEW形式.但是每当要更新视图定义时,常常报"cannot change name of view column xxx to yyy"的错误,通常是在视图修改某字段名.中间增加字段.删除字段时发生. 究其原因,是PostgreSQL虽然支持CREATE OR REPLACE VIEW语义,却有着容易让人忽略的重要限制(O

  • Vue中使用this.$set()如何新增数据,更新视图

    目录 使用this.$set()新增数据,更新视图 描述 简单的讲就是 说说vue.set() (this.$set)用法 使用this.$set()新增数据,更新视图 描述 如果在实例创建之后添加新的属性到实例上,它不会触发视图更新 简单的讲就是 在页面渲染完成之后,对data里的某个数组或对象进行新增删除属性操作是监听不到的,视图不会更新 <div id='app'>     <el-button @click="setinfo">新增属性</el-b

  • vuex state及mapState的基础用法详解

    先使用vue cli构建一个自己的vue项目 1.npm i -g vue-cli 2.vue init webpack sell (sell是你的项目名) 3.一路回车(在这个过程中会提示你是否安装一些依赖包,比如vue-router,es6语法检查等等,这些根据你个人习惯或者癖好选择Y/N) 4.npm i (这个是安装项目的依赖包) 5.npm run dev(启动你的vue项目) 这个时候如果在页面上看到了vue的logo说明你的vue的项目基础构建已经完成,然后你可以删除掉没有用的组件

随机推荐