想了解Vuex?一定先把这篇笔记码住

2023-01-02 0 415

一、Vuex是干嘛用的?

Vuex是用作对繁杂应用领域展开状况管理工作用的(非官方讲法是它是一类状况区域化)。

“坎氏不必宰牛刀”。对单纯的工程项目,显然并不需要Vuex这把“宰牛刀”。那单纯的工程项目用甚么呢?用Vue.js非官方提供更多的“该事件汇流排”就能了。

二、他们import进去的Vuex第一类都包涵些甚么呢?

他们采用Vuex的这时候是不是用呢?一般来说都是这种:

import Vue from vue;

import Vuex from vuex;

Vue.use(Vuex);

new Vuex.Store({

state: { //置放state的值

count: 0,

str:”abcd234″

},

getters: { //放置getters方式

strLen: state => state.str.length

},

// mutations根本无法是同步操作

mutations: { //置放mutations方式

increment(state, payload) {

//在这里改变state中的数据

state.count = payload.number;

}

},

// actions能是异步操作

actions: { //置放actions方式

actionName({ commit }) {

//dosomething

commit(mutationName)

},

getSong ({commit}, id) {

api.getMusicUrlResource(id).then(res => {

let url = res.data.data[0].url;

})

.catch((error) => { // 错误处理

console.log(error);

});

}

}

});

new Vue({

el: #app,

store,

});

这里import进去的Vuex是个甚么东西呢?他们用console.log把它输出一下:

console.log(Vuex)

通过输出,他们发现其结构如下:

想了解Vuex?一定先把这篇笔记码住

可见,import进去的Vuex它实际上是一个第一类,里面包涵了Store这一构造函数,还有几个mapActions、mapGetters、mapMutations、mapState这几个辅助方式(后面再讲)。

除此之外,还有一个install方式。他们发现,import之后要对其展开Vue.use(Vuex)的操作。根据这两个线索,他们就明白了,Vuex本质上就是一个Vue.js的插件。

三、创建好的store实例是不是在各个组件中都能引用到?

Vuex 通过 store 选项,提供更多了一类机制将状况从根组件“注入”到每一个子组件中(需调用 Vue.use(Vuex)):

const app = new Vue({

el: #app,

// 把 store 第一类提供更多给 “store” 选项,这能把 store 的实例注入所有的子组件

store,

components: { Counter },

template: `

`

})

通过在根实例中注册 store 选项,该 store实例会注入到根组件下的所有子组件中,且子组件能通过 this.$store 访问到。

四、Vuex中的几大核心概念

1. State

这个很好理解,就是状况数据。Vuex所管理工作的就是状况,其它的如Actions、Mutations都是来辅助实现对状况的管理工作的。Vue组件要有所变化,也是直接受到State的驱动来变化的。

2. Getters

Getters本质上是用来对状况展开加工处理。Getters与State的关系,就像Vue.js的computed与data的关系。 getter的返回值会根据它的依赖被缓存起来,且只有当它的依赖值发生了改变才会被重新计算。

能通过this.$store.getters.valueName对派生出来的状况展开访问。或者直接采用辅助函数mapGetters将其映射到本地计算属性中去。

3. Mutations

更改 Vuex的 store中的状况的唯一方式是提交 mutation。Vuex中的 mutation非常类似于该事件:每个 mutation都有一个字符串的 该事件类型 (type) 和 一个 回调函数 (handler)。这个回调函数就是他们实际展开状况更改的地方。

你不能直接调用一个 mutation handler。这个选项更像是该事件注册:“当触发一个类型为 increment 的 mutation时,调用此函数。”要唤醒一个mutation handler,你需要以相应的 type 调用store.commit方式,并且它会接受 state 作为第一个参数,也能向 store.commit 传入额外的参数,即 mutation 的 载荷(payload):

const store = new Vuex.Store({

state: {

count: 1

},

mutations: {

increment (state) {

// 变更状况

state.count++

}

}

})

store.commit(increment)

// …

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

})

// ***或者以第一类风格提交***

mutations: {

increment (state, payload) {

state.count += payload.amount

}

}

store.commit({

type: increment,

amount: 10

})

Mutation需遵守 Vue的响应规则

既然 Vuex的 store中的状况是响应式的,那么当他们变更状况时,监视状况的 Vue组件也会自动更新。这也意味着 Vuex中的 mutation也需要与采用 Vue一样遵守一些注意事项:

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

当需要在第一类上添加新属性时,你应该:采用 Vue.set(obj, newProp, 123), 或者

以新第一类替换老第一类。例如,利用第一类展开运算符 他们能这种写:

state.obj = { …state.obj, newProp: 123 }

Mutation必须是同步函数

一条重要的原则就是要记住 mutation必须是同步函数。为甚么?请参考下面的例子:

mutations: {

someMutation (state) {

api.callAsyncMethod(() => {

state.count++

})

}

}

现在想象,他们正在 debug一个 app并且观察 devtool中的 mutation日志。每一条 mutation被记录,devtools都需要捕捉到前一状况和后一状况的快照。然而,在上面的例子中 mutation中的异步函数中的回调让这不可能完成:因为当 mutation触发的这时候,回调函数还没有被调用,devtools不知道甚么这时候回调函数实际上被调用——实质上任何在回调函数中展开的状况的改变都是不可追踪的。

在组件中提交 Mutation

除了这种采用 this.$store.commit(xxx) 提交 mutation的方式之外,还有一类方式,即采用 mapMutations 辅助函数将组件中的 methods映射为 this.$store.commit。例如:

import { mapMutations } from vuex

export default {

// …

methods: {

…mapMutations([

increment, // 将 `this.increment()` 映射为 `this.$store.commit(increment)`

// `mapMutations` 也支持载荷:

incrementBy // 将 `this.incrementBy(amount)` 映射为 `this.$store.commit(incrementBy, amount)`

]),

…mapMutations({

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

})

}

}

经过这种的映射之后,就能通过调用方式的方式来触发其对应的(所映射到的)mutation commit了,比如,上例中调用add()方式,就相当于执行了this.$store.commit(increment)了。

4. Actions

Action类似于 mutation,不同在于:

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

Action能包涵任意异步操作。

Action函数接受一个与 store实例具有相同方式和属性的 context第一类,因此你能调用 context.commit 提交一个

分发 Action

Action 通过 store.dispatch 方法触发:

store.dispatch(increment)

Actions 支持同样的载荷方式和第一类方式展开分发:

// 以载荷形式分发

store.dispatch(incrementAsync, {

amount: 10

})

// 以第一类形式分发

store.dispatch({

type: incrementAsync,

amount: 10

})

另外,你需要知道, this.$store.dispatch 能处理被触发的 action 的处理函数返回的 Promise,并且 this.$store.dispatch 仍旧返回 Promise。

actions: {

actionA ({ commit }) {

return new Promise((resolve, reject) => {

setTimeout(() => {

commit(someMutation)

resolve()

}, 1000)

})

}

}

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

// …

})

5. Module

Module是甚么概念呢?它实际上是对store的一类切割。由于Vuex采用的是单一状况树,这种整个应用领域的所有状况都会集中到一个比较大的第一类上面,那么,当应用领域变得非常繁杂时,store第一类就很可能变得相当臃肿!Vuex允许他们将 store分割成一个个的模块(module)。每个模块拥有自己的 state、mutation、action、getter、甚至是嵌套子模块——从上至下展开同样方式的分割。

(1)模块的局部状况

对每个模块内部的 mutation 和 getter,接收的第一个参数就是模块的局部状况第一类,对模块内部的 getter,根节点状况会作为第三个参数暴露出来。同样,对模块内部的 action,局部状况通过 context.state 暴露出来,根节点状况则为 context.rootState:

const moduleA = {

state: () => ({

count: 0

}),

mutations: {

increment (state) {

// 这里的 `state` 第一类是模块的局部状况

state.count++

}

},

getters: {

doubleCount (state) {

return state.count * 2

},

sumWithRootCount (state, getters, rootState) {

return state.count + rootState.count

}

},

actions: {

incrementIfOddOnRootSum ({ state, commit, rootState }) {

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

commit(increment)

}

}

}

}

(2)命名空间

默认情况下,模块内部的 action、mutation和 getter是注册在全局命名空间的——这种使得多个模块能够对同一 mutation或 action作出响应。

如果希望你的模块具有更高的封装度和复用性,你能通过添加 namespaced: true 的方式使其成为带命名空间的模块。当模块被注册后,它的所有 getter、action及 mutation都会自动根据模块注册的路径调整命名。例如:

const store = new Vuex.Store({

modules: {

account: {

namespaced: true,

// 模块内容(module assets)

state: { … }, // 模块内的状况已经是嵌套的了,采用 `namespaced` 属性不会对其产生影响

getters: {

isAdmin () { … } // -> getters[account/isAdmin]

},

actions: {

login () { … } // -> dispatch(account/login)

},

mutations: {

login () { … } // -> commit(account/login)

},

// 嵌套模块

modules: {

// 继承父模块的命名空间

myPage: {

state: { … },

getters: {

profile () { … } // -> getters[account/profile]

}

},

// 进一步嵌套命名空间

posts: {

namespaced: true,

state: { … },

getters: {

popular () { … } // -> getters[account/posts/popular]

}

}

}

}

}

})

启用了命名空间的 getter 和 action 会收到局部化的 getter,dispatch 和 commit。换言之,

你在采用模块内容(module assets)时不需要在同一模块内额外添加空间名前缀。更改namespaced 属性后不需要修改模块内的代码。

(3)在带命名空间的模块内访问全局内容(Global Assets)

如果你希望采用全局 state和 getter,rootState 和 rootGetters 会作为第三和第四参数传入 getter,也会通过 context 第一类的属性传入 action。

若需要在全局命名空间内分发 action或提交 mutation,将 { root: true } 作为第三参数传给 dispatch 或 commit 即可。

modules: {

foo: {

namespaced: true,

getters: {

// 在这个模块的 getter 中,`getters` 被局部化了

// 你能采用 getter 的第四个参数来调用 `rootGetters`

someGetter (state, getters, rootState, rootGetters) {

getters.someOtherGetter // -> foo/someOtherGetter

rootGetters.someOtherGetter // -> someOtherGetter

},

someOtherGetter: state => { … }

},

actions: {

// 在这个模块中, dispatch 和 commit 也被局部化了

// 他们能接受 `root` 属性以访问根 dispatch 或 commit

someAction ({ dispatch, commit, getters, rootGetters }) {

getters.someGetter // -> foo/someGetter

rootGetters.someGetter // -> someGetter

dispatch(someOtherAction) // -> foo/someOtherAction

dispatch(someOtherAction, null, { root: true }) // -> someOtherAction

commit(someMutation) // -> foo/someMutation

commit(someMutation, null, { root: true }) // -> someMutation

},

someOtherAction (ctx, payload) { … }

}

}

}

(4)在带命名空间的模块注册全局 action

若需要在带命名空间的模块注册全局 action,你可添加 root: true,并将这个 action 的定义放在函数 handler 中。例如:

{

actions: {

someOtherAction ({dispatch}) {

dispatch(someAction)

}

},

modules: {

foo: {

namespaced: true,

actions: {

someAction: {

root: true,

handler (namespacedContext, payload) { … } // -> someAction

}

}

}

}

}

(5)带命名空间绑定函数

当采用 mapState, mapGetters, mapActions 和 mapMutations 这些函数来绑定带命名空间的模块时,写起来可能比较繁琐:

computed: {

…mapState({

a: state => state.some.nested.module.a,

b: state => state.some.nested.module.b

})

},

methods: {

…mapActions([

some/nested/module/foo, // -> this[some/nested/module/foo]()

some/nested/module/bar // -> this[some/nested/module/bar]()

])

}

对这种情况,你能将模块的空间名称字符串作为第一个参数传递给上述函数,这种所有绑定都会自动将该模块作为上下文。于是上面的例子能简化为:

computed: {

…mapState(some/nested/module, {

a: state => state.a,

b: state => state.b

})

},

methods: {

…mapActions(some/nested/module, [

foo, // -> this.foo()

bar // -> this.bar()

])

}

(6)模块动态注册

在 store创建之后,你能采用 store.registerModule 方式注册模块:

import Vuex from vuex

const store = new Vuex.Store({ /* 选项 */ })

// 注册模块 `myModule`

store.registerModule(myModule, {

// …

})

// 注册嵌套模块 `nested/myModule`

store.registerModule([nested, myModule], {

// …

})

之后就能通过 store.state.myModule 和

store.state.nested.myModule 访问模块的状况。

模块动态注册功能使得其他 Vue 插件能通过在 store 中附加新模块的方式来采用 Vuex 管理工作状况。例如,vuex-router-sync插件就是通过动态注册模块将vue-router 和 vuex结合在一起,实现应用领域的路由状况管理工作。

你也能采用 store.unregisterModule(moduleName) 来动态卸载模块。注意,你不能采用此方式卸载静态模块(即创建 store时声明的模块)。

注意,你能通过 store.hasModule(moduleName) 方式检查该模块是否已经被注册到 store。

保留 state

在注册一个新 module时,你很有可能想保留过去的 state,例如从一个服务端渲染的应用领域保留 state。你能通过 preserveState 选项将其归档:store.registerModule(a, module, { preserveState: true })。

当你设置 preserveState: true 时,该模块会被注册,action、mutation和 getter会被添加到 store中,但是 state不会。这里假设 store的 state已经包涵了这个 module的 state并且你不希望将其覆写。

(7)模块重用

有时他们可能需要创建一个模块的多个实例,例如:

创建多个 store,他们共用同一个模块 (例如当 runInNewContext 选项是 false 或 once 时,为了在服务端渲染中避免有状况的单例 (opens new window))

在一个 store中多次注册同一个模块

如果他们采用一个纯第一类来声明模块的状况,那么这个状况第一类会通过引用被共享,导致状况第一类被修改时 store 或模块间数据互相污染的问题。

实际上这和 Vue组件内的 data 是同样的问题。因此解决办法也是相同的——采用一个函数来声明模块状况(仅 2.3.0+ 支持):

const MyReusableModule = {

state: () => ({

foo: bar

}),

// mutation, action 和 getter 等等…

}

五、Vuex中的表单处理当在严格模式中采用 Vuex时,在属于 Vuex的 state上采用 v-model 会比较棘手:

<input v-model=”obj.message”>

假设这里的 obj 是在计算属性中返回的一个属于 Vuex store 的第一类,在用户输入时,v-model 会试图直接修改 obj.message。在严格模式中,由于这个修改不是在 mutation 函数中执行的, 这里会抛出一个错误。

用“Vuex 的思维”去解决这个问题的方式是:给 <input> 中绑定 value,然后侦听 input 或者 change 该事件,在该事件回调中调用一个方式:

<input :value=”message” @input=”updateMessage”>

computed: {

…mapState({

message: state => state.obj.message

})

},

methods: {

updateMessage (e) {

this.$store.commit(updateMessage, e.target.value)

}

}

下面是 mutation函数:

// …

mutations: {

updateMessage (state, message) {

state.obj.message = message

}

}

必须承认,这种做比单纯地采用“v-model + 局部状况”要啰嗦得多,并且也损失了一些 v-model 中很有用的特性。另一个方式是采用带有 setter 的双向绑定计算属性:

<input v-model=”message”>

computed: {

message: {

get () {

return this.$store.state.obj.message

},

set (value) {

this.$store.commit(updateMessage, value)

}

}

}

云管理工作服务专家新钛云服 林泓辉原创

相关文章

发表评论
暂无评论
官方客服团队

为您解决烦忧 - 24小时在线 专业服务