vuex 使用说明

安装

npm

npm install vuex --save

在一个模块化的打包系统中,您必须显式地通过Vue.use()来安装Vuex。

import Vue from 'vue'

import Vuex from 'vuex'

Vue.use(Vuex)

Vuex是一个专为Vue.js应用程序开发 的状态管理模式,集中式存储管理应用的所有组件状态。

状态管理包含以下几个部分状态:

state驱动应用的数据源;

view以生命方式将state映射到视图。

actions响应在view上的用户书输入导致的状态变化。

帮助我们管理共享状态,中大型单页面应用。

state

单一状态树,Vuex使用单一状态树用一个对象就包含了全部的应用层级状态。

在Vue组件中获得Vuex状态。

由于Vuex的状态存储是响应式的,从store实例中读取状态最简单的方法

就是在计算属性中返回某个状态。

创建一个Counter组件

const Counter = {

template: '

{{ count }}
'

computed: {

count (){

return store.state.count

}

}

每次store.state.count变化的时候,都会重新求取计算属性,并且触发更

新相关的DOM

Vuex通过store选项,提供了一种机制将状态从根组件『注入』到每一个子组件

中(需调用Vue.use(Vuex)):

const app = new Vue({

el:'#app',

//把store对象提供给“store”选项,这可以把store的实例注入所有的子组件

store,

components: {Counter},

template: '

'

})

通过在根实例中注册store选项,该store实例会注册入到跟组件下的所有

子组件,且子组件能通过this.$store访问到。更新counter的实现:

const Counter = {

template : '

{{ count }}
',

computed: {

count this.$store.state.count

}

}

mapState辅助函数

当一个组件需要获取多个状态时候,将这些状态都声明为计算属性会有些冗余。

为了解决这个问题,我们可以使用mapState辅助函数帮助我们生成计算属性。

//在单独构建的版本中辅助函数为Vuex.mapState

import { mapState } from 'vuex'

export default {

computed: mapState({

//箭头函数可以使代码更简洁

count: state => state.count,

//传字符串参数‘count’等同于‘state => state.count’

countAlias: 'count',

//为了能够使用‘this’获取局部状态,必须使用常规函数

countPlusLocalState(state) {

return state.count + this.localCount

}

})

}

当映射的计算属性的名称与state的子节点名称相同时,我们也

可以给mapState传一个字符串数组。

computed: mapState([

//映射this.count为store.state.count

'count'

])

组件仍然保有局部状态。

Getters

有时候我们需要从store中的state中 的state中派生一些状态,列如对列表进

行过滤并计算。

computed: {

doneTodosCount() {

return this.$store.state.todos.filter(todo => todo.done).length

}

}

Vuex允许我们再store中定义getters (可以认为是stroe的计算属性)

Getters接受state作为其第一个参数。

const store = new Vuex.Store({

state: {

todos:[

{id:1, text: '...' ,done: true},

{id:2,text:'...',done: false}

]

},

getters: {

doneTodos: state => {

return state.todos.filter(todo=> todo.done)

}

}

})

Getters会暴露为store.getters对象:

store.getters.doneTodos // [{id:1,text: '...',done:true}]

Getter也可以接受其他getters作为第二个参数:

getters: {

doneTodosCount: (state,getters) => {

return getters.doneTodos.length

}

}

store.getters.doneTodosCount  // -> 1

我们可很容易的在任何组件中使用

computed: {

doneTodosCount() {

return this.$store.getters.doneTodosCount

}

}

mapGetters辅助函数

mapGetters辅助函数仅仅是store中的getters映射到局部计算属性。

import {mapGetter} form 'vuex'

export default {

computed: {

//使用对象展开运算符将getters混入

...mapGetters([

‘doneTodosCount’,

'anotherGetter'

])

}

}

如果你想讲一个getter属性另取名字,使用对象性时

mapGetters({

//映射this.doneCount为store.getters.doneTodosCount

doneCount: 'doneTodosCount'

})

Mutations

更改Vuex的store中的状态的唯一方式就是提交mutation Vuex中的mutation

非常类似于事件,每个mutation都有一个字符串的 事件类型 和回调函数。这个

回调函数就是我们实际进行状态更改的地方。并且他会接受state作为第一个参数。

const store = new Vue.Store({

state: {

count: 1

},

mutations: {

inctement (state) {

state.count++

}

}

})

当触发一个类型为increment的mutation时,调用此函数。”要唤醒一个

mutation handler,你需要以相应的type调用store.commit方法

store.commit('increment')

提交载荷(Payload)

你可以向store.commit传入额外的参数,即mutation的载荷:

mutations: {

increment (state, n) {

state.count += n

}

}

store.commit('increment', 10)

在大多数情况下,载荷应该是一个对象,这样可以包含多个字段并且记录mutation会更易读。

mutations: {

increment (state,payload) {

state.count += payload.amount

}

}

store.commit('increment', {

amount:10

})

对象风格的提交方式

提交mutation的另一种方式直接使用包含type属性的对象:

store.commit({

type: 'increment',

amount:10

})

当使用对象风格的提交方式,整个对象作为载荷传给mutation函数,因此handler保持不变:

mutations: {

increment (state, payload) {

state.count += payload.amount

}

}

Mutations需遵守vue的响应规则

既然Vuex的store中的状态是响应式的,那么当我们变更状态时,监视状态的vue更新,这也意味值Vue中的mutation也需要与使用Vue一样遵守一些注意事项。

1.最好提前在你的store中初始化好所有的所需要的属性。

2.当需要在对象上提交新属性时,你应该使用

Vue.set(obj, 'newProp', 123)

使用新对象代替老对象state.obj= {...state.obj ,newProp: 123}

使用常量替代Mutation事件类型

使用常量替代mutation事件类型在各种Flux实现中是很常见的模式

export const SOME_MUTATION = 'SOME_MUTATION';

import Vuex from 'vuex'

import {SOME_MUTATION } from './mutation-types'

const store = new Vuex.Store({

state:{...}

mutations: {

//我们可以使用ES2015风格的计算属性命名功能来使用一个常量作为函数名

[SOME_MUTATION] (state) {

//  mutate state

}

}

})

mutation必须是同步函数

一条重要的原则是记住mutation必须是同步函数。

mutations: {

someMutation (state) {

api.callAsyncMethod(() => {

state.count++

})

}

}

在组件中提交Mutations

你可以在组件中使用this.$store.commit('xxx')提交mutation,或者使使用mapMutations辅助函数将组建中的methods映射为store.commit调用(需要在根节点注入store)

import {mapMutations} from 'vuex'

expor default {

methods: {

mapMutations([

methods: {

mapMutations([

'increment'  //映射this.increment()为this.$store.commit('increment')

]),

mapMutations({

add: 'increment'  //映射this.add()为this.$store.commit('increment')

})

}

])

}

}

Actions

在mutation中混异步调用会导致你的程序很难调试。

Actions

Action类似于mutation,不同在于。

Action提交的是mutation ,而不是直接变更状态。

Action可以包含任意异步操作。

注册一个简单的action

const store = new Vuex.Store({

state: {

count:0

},

mutations: {

increment (state) {

state.count++

}

},

actions: {

increment (context){

context.commit('increment')

}

}

})

Action函数接受一个与store实例具有相同方法和属性的context对象,因此你可以调用context.commit提交一个mutation,或者通过context.state和

context.getters来获取state和getters当我们在之后介绍到Modules时,

你就知道context对象为什么不是store实例本身了。

actions: {

increment({commit}){

commit('increment')

}

}

分发Action

Action通过store.dispatch方法触发:

store.dispatch('increment')

我们可以在action内部执行异步操作。

actions: {

incrementAsync({commit}){

setTimeout(() => {

commit('increment')

},1000)

}

}

Actions支持同样的载荷方式和对象方式进行分发

//以载荷形式分发

store.dispatch('incrementAsync',{

amount:10

})

//以对象形式分发

store.dispatch({

type: 'incrementAsync',

amount:10

})

在组件中分发Action

你在组件中使用this.$store.dispatch('xxx')分发action,或者使用map Actions辅助函数将组件的methods映射为store.dispatch调用

import {mapActions } from 'vuex'

export default{

methods:([

'increment'  //映射this.increment()为this.$store.dispatch('increment')

])

mapActions({

add: 'inctement'    //映射this.add()为this.$store.dispatch('increment')

})

}

组合Actions

Action通常是异步的,那么如何知道action什么时候结束。

你需要明白store.dispatch可以处理被处触发的action的回调函数返回的Promise

并且store.dispatch仍旧返回Promise

actions: {

actionA({commit}){

return new Promise((resolve)=>{

setTimeout (() => {

commit('someMutation')

resolve()

},1000)

})

}

}

现在你可以

store.dispatch('actionA').then(()=>{

//...

})

在另一个action中也可以

actions: {

actionB({dispath,commit}){

return dispatch('actionA').then(() => {

commit('someOtherMutation')

})

}

}

我们利用async/ await

//假设getData()和getOther()返回的是一个Promis

actions:{

async actionA ({commit}){

commit('gotData',await getData())

},

async actionB({dispatch,commit}){

await dispatch('actionA')  //等待actionA完成

commit('goOtherData', await getOtherData())

}

}

Modules

使用单一状态树,当应用变的很大的时候,store对象会变的臃肿不堪。

Vuex允许我们将store分割到模块。每一个模块都有自己的state, mutation,action, getters,甚至是嵌套子模块从上到下进行类似的分割。

const moduleA = {

state: {...},

mutations: {...}

actions: {...}

getters:{...}

}

const moduleA = {

state: {...},

mutations: {...}

actions: {...}

}

const store = new Vuex.Store({

modules: {

a:moduleA,

b:moduleB

}

})

store.state.a   // -> moduleA的状态

store.state.b // -> moduleB的状态

模块的局部状态

对于模块内部的mutation和getter,接收的第一个参数是模块的局部状态。

const moduleA = {

state: {count:0},

mutations: {

increment (state) {

//  state模块的局部状态

state.count++

}

},

getters: {

doubleCount (state) {

return state.count * 2

}

}

}

同样对于模块内部的action, context.state是局部状态,根节点的窗台石context.rootState:

const moduleA = {

actions: {

incrementIfOddOnRootSum ({state, commit ,rootState}) {

if((state.count + rootState.count) %2 ===1){

commit('increment')

}

}

}

}

对于模块内部的getter,跟节点状态会作为第三个参数:

const  moduleA = {

getters: {

getters: {

sumWithRootCount (state,getters,rootState) {

return state.count + rootState.count

}

}

}

}

命名空间

模块内部的action, mutation ,和getter现在仍然注册在全局命名空间    这样保证了多个模块能够响应同一mutation或action.也可以通过添加前缀 或者 后缀的

方式隔离各个模块,以免冲突。

//定义getter, action ,和mutation的名称为常量,以模块名‘todo’为前缀。

export  const  DONE_COUNT = 'todos/DONE_COUNT'

export  const  FETCH_ALL =  'todos/FETCH_ALL'

export  const TOGGLE_DONE = 'todos/TOGGLE_DONE'

import * as types form '../types'

//使用添加了解前缀的名称定义,getter, action和mutation

const todosModule = {

state : {todo: []},

getters: {

[type.DONE_COUNT]  (state) {

}

}

actions: {

[types.FETCH_ALL] (context,payload) {

}

},

mutations: {

[type.TOGGLE_DONE] (state, payload)

}

}

模块动态注册

在store创建之后,你可以使用store.registerModule方法注册模块。

store.registerModule('myModule',{})

模块的状态将是store.state.myModule.

模块动态注册功能可以使让其他Vue插件为了应用的store附加新模块

以此来分割Vuex的状态管理。

项目结构

Vuex并不限制你的代码结构。但是它规定了一些需要遵守的规则:

1.应用层级的状态应该集中到单个store对象中。

2.提交mutation是更改状态的唯一方法,并且这个过程是同步的。

3.异步逻辑应该封装到action里面。

只要你遵守以上规则,如何组织代码随你便。如果你的store文件太大,

只需将action、mutation、和getters分割到单独的文件

对于大型应用,我们会希望把Vuex相关代码分割到模块中。下面是项目结构示例

├── index.html

├── main.js

├── api │

└── ... #抽取出API请求

├── components

│   ├── App.vue

│   └── ...

└── store

├── index.js     #我们组装模块并导出store的地方

├── actions.js        #根级别的action

├── mutations.js      #根级别的mutation

└── modules

├── cart.js       #购物车模块

└── products.js   #产品模块

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,817评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,329评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,354评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,498评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,600评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,829评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,979评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,722评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,189评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,519评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,654评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,329评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,940评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,762评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,993评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,382评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,543评论 2 349

推荐阅读更多精彩内容

  • Vuex是什么? Vuex 是一个专为 Vue.js应用程序开发的状态管理模式。它采用集中式存储管理应用的所有组件...
    萧玄辞阅读 3,111评论 0 6
  • Vuex 的学习记录 资料参考网址Vuex中文官网Vuex项目结构示例 -- 购物车Vuex 通俗版教程Nuxt....
    流云012阅读 1,452评论 0 7
  • import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(V...
    F_imok阅读 2,646评论 0 12
  • vuex 场景重现:一个用户在注册页面注册了手机号码,跳转到登录页面也想拿到这个手机号码,你可以通过vue的组件化...
    sunny519111阅读 8,009评论 4 111
  • Vuex 是什么? ** 官方解释:Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式**。它采用集中...
    Rz______阅读 2,300评论 1 10