一、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)
通过输出,他们发现其结构如下:
可见,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)
}
}
}云管理工作服务专家新钛云服 林泓辉原创