詳解探索 vuex 2.0 以及使用 vuejs 2.0 + vuex 2.0 構(gòu)建記事本應(yīng)用
前言
首先說明這并不是一個教程貼,而記事本應(yīng)用是網(wǎng)上早有的案例,對于學(xué)習 vuex 非常有幫助。我的目的是探索 vuex 2.0 ,然后使用 vuejs 2.0 + vuex 2.0 重寫這個應(yīng)用,其中最大的問題是使用 vue-cli 構(gòu)建應(yīng)用時遇到的問題。通過這些問題深入探索 vue 以及 vuex 。
我對于框架的學(xué)習一直斷斷續(xù)續(xù),最先接觸的是 react,所以有一些先入為主的觀念,喜歡 react 更多一點,尤其在應(yīng)用的構(gòu)建層面來說。之所以斷斷續(xù)續(xù),是因為自己 JS 基礎(chǔ)較弱,剛開始學(xué)習的時候,只是比著葫蘆畫瓢,雖然可以做出點東西,但對于其中的一些概念仍然云里霧里,不知所云,無法深入理解框架。所以我又臨時放棄框架的學(xué)習,開始學(xué)習 JS 基礎(chǔ)。事實證明打牢基礎(chǔ)之后,學(xué)習框架以及理解框架是神速的。而學(xué)習 webgl 和 three.js 的過程與此類似。沒有 webgl 的基礎(chǔ),學(xué)習 three.js 只會停留在初級階段。
我在過去的半年參加了很多面試,幾乎無一例外的都會被問框架的使用情況,但是其中很多公司屬于隨波逐流,使用框架比較盲目。甚至覺得使用框架是極其高大上的事情。雖然我學(xué)習過框架,但畢竟沒有深入學(xué)習也沒有拿得出手的項目,所以只是只言片語的說兩句,大部分知識是懵懂的。然而面對面試官不屑的神情以及以此作為選拔的指標,心想這樣的面試官太膚淺。當然很多公司的面試還是以基礎(chǔ)為主??蚣軐儆谔剿鳎ハ鄬W(xué)習的狀態(tài)。我在這篇文章中強調(diào)一點,學(xué)習能力以及解決問題的能力更重要。
開始吧
言歸正傳,對于這個筆記本案例,大家可以直接百度搜 vue notes ,這是一篇英文教程,大家看到的都是翻譯的。在剛開始 vue 資料稀缺的時候,這樣的文章非常珍貴。demo 點這里。說白了,算是 todoMVC 案例的一個變體。當初覺得這個例子非常好,想跟著學(xué)一學(xué),結(jié)果一拖半年過去了。這幾天終于抽時間把這個例子敲了一遍。學(xué)習在于舉一反三。如果大家按照網(wǎng)上教程來做,那么 NPM 包默認安裝的都是最新版本,運行會報錯。所以如果用 vuex 2 要怎么寫呢?
以下是 notes-vuex-app 的源文件目錄:

在使用 vue 2 重寫這個 app 之前,我在想能不能不改變文件目錄結(jié)構(gòu)以及配置位置呢?就是用比較生硬的方式重寫,或者說單純的語法修改。事實是可行的,否則我就不會寫這篇文章了。然而面對的問題非常多,但卻因此深入的理解了 vue 以及 vuex。最大的問題是 webpack 的構(gòu)建,如果使用 webpack 2.0+的話,坑比較多。本人是菜鳥,所以最終選擇了 vue-cli 提供的兩個 webpack 的模板,分別是 webpack-simple 和 webpack,我先使用 webpack-simple,它和原 app 的結(jié)構(gòu)基本吻合。目錄如下:

使用 vue-cli 生成基本目錄之后,再安裝 vuex2 。
main.js 的小改動
原示例 main.js 如下所示,但運行出錯了,主要是 Vue 2 的根實例渲染稍有變化
import Vue from 'vue'
import store from './vuex/store'
import App from './components/App.vue'
new Vue({
store, // 注入到所有子組件
el: 'body',
components: { App }
})
改正之后:
import Vue from 'vue'
import store from './vuex/store'
import App from './components/App.vue'
new Vue({
store, // inject store to all children
el: '#app',
template: '<App/>',
components: { App }
})
或者
import Vue from 'vue'
import store from './vuex/store'
import App from './components/App.vue'
new Vue({
store, // inject store to all children
el: '#app',
render: h => h(App)
})
vuex 2 的變化
這個應(yīng)用改寫的主要問題集中在 vuex 2 的變化上,這些變化確實會讓人感到凌亂,我無數(shù)次抓耳撓腮的罵娘。不過通過官方給出的示例也可以看出一些端倪。
首先是 action.js,只需注意一點,所有的 dispatch 都要改成 commit。
export const addNote = ({ commit }) => {
commit('ADD_NOTE')
}
export const editNote = ({ commit }, e) => {
commit('EDIT_NOTE', e.target.value)
}
export const deleteNote = ({ commit }) => {
commit('DELETE_NOTE')
}
export const updateActiveNote = ({ commit }, note) => {
commit('SET_ACTIVE_NOTE', note)
}
export const toggleFavorite = ({ commit }) => {
commit('TOGGLE_FAVORITE')
}
store.js 變化也不大,但是要注意幾個地方:
import Vue from 'vue'
import Vuex from 'vuex'
import * as actions from './actions'
Vue.use(Vuex)
const state = {
notes: [],
activeNote: {}
}
const mutations = {
ADD_NOTE (state) {
const newNote = {
text: 'New note',
favorite: false
}
state.notes.push(newNote)
state.activeNote = newNote
},
EDIT_NOTE (state, text) {
state.activeNote.text = text
},
DELETE_NOTE (state) {
state.notes.splice(state.notes.indexOf(state.activeNote),1)
state.activeNote = state.notes[0] || {}
},
TOGGLE_FAVORITE (state) {
state.activeNote.favorite = !state.activeNote.favorite
},
SET_ACTIVE_NOTE (state, note) {
state.activeNote = note
}
}
const getters = {
notes: state => state.notes,
activeNote: state => state.activeNote,
activeNoteText: state => state.activeNote.text
}
export default new Vuex.Store({
state,
mutations,
actions,
getters
})
原示例文件中沒有將 getters 寫到 store.js 中,而是直接在組件中定義的。為了更清晰,我仿照官方示例也提取出來寫在了 store.js 中,這樣在組件中調(diào)用時比較方便。其次也引入了 action.js,并作為 actions 對象傳遞給 Vuex.store(),這算是 vuex 的標準寫法吧,對于后面在組件中調(diào)用比較有利。
其中要注意 DELETE_NOTE (state){} 這個方法,原示例使用了 vue1 提供的 remove 方法,但是 vue2 中去掉了這個方法。仔細想想就會明白,這個函數(shù)的作用就是刪除 notes 數(shù)組中的元素??梢允褂迷?splice 方法。如果 JS 基礎(chǔ)扎實的話,這里應(yīng)該很好理解,沒有什么大問題。其次相比原示例,添加一個刪除后操作的判斷。
我之前一直不太理解 flux 的概念,感覺像是新東西,完全不知道它的目的及作用。換成 Vuex,還是有點稀里糊涂。但是通過修改這個示例,基本算是開竅了。這些東西本身并沒有玄機奧妙,想一想,如果我們不用框架,而是自己手寫一個 todoMVC 時要怎么做?應(yīng)該也是這樣的思路,定義一個 notes 數(shù)組變量以及 activeNote 的變量。然后在創(chuàng)建一些改變狀態(tài)的方法。我在面試中遇到過一個情況,面試官反復(fù)問我為什么需要使用框架,用 jQuery 不是也可以實現(xiàn)嗎?這樣說確實沒錯,用比較原始的方法當然可以做,只是代碼結(jié)構(gòu)會冗余或者凌亂,缺少小而美的特點??蚣芤约霸O(shè)計模式對代碼做了整合封裝,對于一個 CURD 應(yīng)用比較友好,實現(xiàn)起來更方便更簡單。我對于 Vuex 的理解就是,它是一個對象,封裝了與狀態(tài)相關(guān)的方法和屬性。而所謂的狀態(tài)就是點擊、按鍵等操作之后的變化。
組件中使用 vuex
先看一下 Toolbar.vue 這個組件。修改后的代碼如下:
<template>
<div id="toolbar">
<i @click="addNote" class="glyphicon glyphicon-plus"></i>
<i @click="toggleFavorite"
class="glyphicon glyphicon-star"
:class="{starred: activeNote.favorite}"></i>
<i @click="deleteNote" class="glyphicon glyphicon-remove"></i>
</div>
</template>
<script>
import { mapGetters, mapActions } from 'vuex'
export default {
computed: mapGetters([
'activeNote'
]),
methods: {
...mapActions([
'addNote',
'deleteNote',
'toggleFavorite'
])
}
}
</script>
通過和原示例代碼對比,這里的區(qū)別一目了然。我通過在控制臺打印 Vue 實例,折騰很長時間才大體明白怎么回事。vuex 1 在組件中使用時會直接將 getters 以及 actions 掛到 vuex 這個屬性上,沒有提供 mapGetters 及 mapActions 等一些方法。而 vuex2 使用 mapGetters 及 mapActions 等一些方法將 actions 的方法掛到 Vue 實例上??偟膩碚f,都是把 actions 的方法掛到 Vue 實例上。我們從這個層面上談?wù)?Vue 實例,Vue 2 的變化就是其屬性的變化。比如 Vue1 中在 methods 中添加的方法可以在 vue 實例的 $options 屬性中查看,而 vue2 中這些方法可以直接在第一級屬性中查找或者在 $options 屬性下的原型方法中 __proto__ 尋找。在 vue1 中可以查看 vuex 這個屬性,但是 vue2 中移除了。至于其它的不同,大家可以自己對比,通過這種方式,可以深入理解 vue 的設(shè)計思想。
下圖是 Vue1 實例截圖:

ES5 實現(xiàn)擴展運算符
假設(shè)其它組件都以這種方式改好了,就在我們滿心歡喜地運行示例時,又報錯了。問題出在擴展運算符 ... 上,webpack-simple 這個模板無法解析 ES6 的 ...。為此,我又折騰了很久,想試著修改 webpack 的配置文件,但改動太大。我妥協(xié)了,決定拋棄擴展運算符,手寫這個方法。當然如果使用 webpack 的模板就沒有問題,這個比較簡單,我們最后再說。
手寫擴展運算符 ... 之前,我們先看一下 mapActions 這個方法。對于 mapGetters 以及 mapActions 這兩個函數(shù),最簡單的理解辦法就是查看 vuex 的源碼,最終返回的是一個對象。也就是根據(jù)需要獲取 store.js 中 actions 對象的某些方法。然后通過擴展運算符把返回的對象拆開然后掛到 Vue 實例上。舉例來說(以下只是擴展運算符的用法之一,別的用法可以參考其它的文章):
var obj = {
a: 1,
b: 2,
}
var methods = {
...obj
}
// console.log(methods)
{
a: 1,
b: 2
}
明白擴展運算符的用法之后就好辦了。為了簡單一點,我直接使用 babel 官網(wǎng)的在線解析器,查看擴展運算符的 ES5 寫法。
// ES5 實現(xiàn)擴展運算符...
var _extends = Object.assign || function(target) {
for (var i = 1; i < arguments.length; i++) {
var source = arguments[i];
for (var key in source) {
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
完整的 Toolbar.vue 組件代碼如下:
<template>
<div id="toolbar">
<i @click="addNote" class="glyphicon glyphicon-plus"></i>
<i @click="toggleFavorite"
class="glyphicon glyphicon-star"
:class="{starred: activeNote.favorite}"></i>
<i @click="deleteNote" class="glyphicon glyphicon-remove"></i>
<i @click="_test" class="glyphicon glyphicon-remove"></i>
</div>
</template>
<script>
import { mapGetters, mapActions } from 'vuex'
// ES5 實現(xiàn)擴展運算符...
var _extends = Object.assign || function(target) {
for (var i = 1; i < arguments.length; i++) {
var source = arguments[i];
for (var key in source) {
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
var actions = mapActions([
'addNote',
'deleteNote',
'toggleFavorite'
]);
var methodsObj = _extends({},actions)
export default {
computed: mapGetters([
'activeNote'
]),
methods:methodsObj
}
</script>
其余兩個子組件類似,相信大家已經(jīng)明白了我的思路,具體代碼如下:
NotesList.vue
<template>
<div id="notes-list">
<div id="list-header">
<h2>Notes | coligo</h2>
<div class="btn-group btn-group-justified" role="group">
<!-- All Notes button -->
<div class="btn-group" role="group">
<button type="button" class="btn btn-default"
@click="show = 'all'"
:class="{active: show === 'all'}">
All Notes
</button>
</div>
<!-- Favorites Button -->
<div class="btn-group" role="group">
<button type="button" class="btn btn-default"
@click="show = 'favorites'"
:class="{active: show === 'favorites'}">
Favorites
</button>
</div>
</div>
</div>
<!-- render notes in a list -->
<div class="container">
<div class="list-group">
<a v-for="note in filteredNotes"
class="list-group-item" href="#" rel="external nofollow"
:class="{active: activeNote === note}"
@click="updateActiveNote(note)">
<h4 class="list-group-item-heading">
{{note.text.trim().substring(0, 30)}}
</h4>
</a>
</div>
</div>
</div>
</template>
<script>
import { mapGetters, mapActions } from 'vuex'
// ES5 實現(xiàn)擴展運算符...
var _extends = Object.assign || function(target) {
for (var i = 1; i < arguments.length; i++) {
var source = arguments[i];
for (var key in source) {
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
var getters = mapGetters([
'activeNote'
]);
var filters = {
filteredNotes: function () {
if (this.show === 'all'){
return this.$store.state.notes
} else if (this.show === 'favorites') {
return this.$store.state.notes.filter(note => note.favorite)
}
}
}
var actions = mapActions(['updateActiveNote'])
var computedObj = _extends({},getters,filters);
var methodsObj = _extends({},actions);
export default {
data () {
return {
show: 'all'
}
},
computed:computedObj,
methods:methodsObj
}
</script>
Editor.vue
<template>
<div id="note-editor">
<textarea
:value="activeNoteText"
@input="editNote"
class="form-control">
</textarea>
</div>
</template>
<script>
import { mapGetters, mapActions } from 'vuex'
export default {
computed:mapGetters(['activeNoteText']),
methods:mapActions(['editNote'])
}
</script>
Webpack 模板
直接使用 vue-cli 的 webpack 模板就會簡單很多,可以直接解析擴展運算符,代碼也會比較簡潔。我就不多說了,直接貼上 github 的地址,大家有不懂的可以看一下:https://github.com/nzbin/notes-app-vuejs2-vuex2
總結(jié)
終于寫完了這篇文章,感慨頗多。這個例子比較典型,學(xué)習的人很多,可能我并不是第一個重寫這個案例的人,我只是與大家分享我的一些心得。順便提一句,為了重寫這個示例并解決遇到的這些小問題,我們可能要使用很多資源,比如 github、codePen、stackoverflow、npm 官網(wǎng)、babel 官網(wǎng)、vuejs 官網(wǎng)、vuex 官網(wǎng)、博客等等。回頭再想想 Vue 到底是什么,一個對象,沒錯,一個集合了很多屬性和方法的對象。為什么要強調(diào)面向?qū)ο蟮闹匾?,可能這就是最好的闡釋,包括 jQuery、react、其它框架等等。一旦遇到問題,在控制臺打印 Vue 實例,反復(fù)查看其屬性可能很有幫助。
最后發(fā)個預(yù)告,下一篇文章我想探討一下面向?qū)ο蟮?CSS,分析幾個優(yōu)秀的 UI 框架,我相信每個人都可以書寫屬于自己的 CSS 框架。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Vue.js在數(shù)組中插入重復(fù)數(shù)據(jù)的實現(xiàn)代碼
這篇文章主要介紹了Vue.js在數(shù)組中插入重復(fù)數(shù)據(jù)的實現(xiàn)代碼,需要的朋友可以參考下2017-11-11
Vue報錯:Uncaught TypeError: Cannot assign to read only propert
這篇文章主要給大家介紹了關(guān)于Vue報錯:Uncaught TypeError: Cannot assign to read only property 'exports' of object '#<Object>' 的解決方法,文中介紹的非常詳細,需要的朋友們下面來一起看看吧。2017-06-06
vue3中關(guān)于路由hash與History的設(shè)置
這篇文章主要介紹了vue3中關(guān)于路由hash與History的設(shè)置方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08
解決vue-quill-editor上傳內(nèi)容由于圖片是base64的導(dǎo)致字符太長的問題
vue-quill-editor默認插入圖片是直接將圖片轉(zhuǎn)為base64再放入內(nèi)容中,如果圖片較多,篇幅太長,就會比較煩惱,接下來通過本文給大家介紹vue-quill-editor上傳內(nèi)容由于圖片是base64的導(dǎo)致字符太長的問題及解決方法,需要的朋友可以參考下2018-08-08
vue3點擊出現(xiàn)彈窗后背景變暗且不可操作的實現(xiàn)代碼
這篇文章主要介紹了vue3點擊出現(xiàn)彈窗后背景變暗且不可操作的實現(xiàn)代碼,本文通過實例代碼圖文相結(jié)合給大家介紹的非常詳細,對大家的學(xué)習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-08-08

