【Vue】狀態(tài)管理

大型單頁面應(yīng)用需要Vuex管理全局/模塊的狀態(tài)擅这,大型單頁面組件如果靠事件(events)/屬性(props)通訊傳值會(huì)把各個(gè)組件耦合在一起。

快速入門


  1. 安裝vuex庫
cnpm install -S vuex
  1. 創(chuàng)建Vuex.Store
import Vue from 'vue'
import Vuex from 'vuex'

Vue.use(Vuex);

const store = new Vuex.Store({
    //組件數(shù)據(jù)源景鼠,單一的state屬性
    state: {
        clickCount: 0
    },
    //相當(dāng)于屬性仲翎,封裝獲取state
    getters: {
        getClickCount: state => {
            return state.clickCount;
        }
    },
    //封裝引起狀態(tài)變化的方法
    mutations: {
        increment(state) {
            state.clickCount++;
        }
    },
    //類似于 mutation痹扇,不同在于actions支持異步,不是直接變更狀態(tài)溯香,而是提交到mutation
    actions: {
        increment(context) {
            context.commit('increment')
        },
        async incrementAsync({ commit }) {
            return new Promise((resolve, reject) => {
                setTimeout(() => {
                    try {
                        commit('increment');
                        resolve(new Date().getTime() + '  成功執(zhí)行');
                    } catch (e) {
                        reject(e);
                    }
                }, 1000)
            });
        }
    }
});
export default store;
  1. Vue實(shí)例加入store
new Vue({
    router: router,
    store: store,
    render: h => h(App),
}).$mount('#app')
  1. 組件獲取store值
<script>
import { mapGetters } from "vuex";

export default {
  computed: mapGetters({ count: ["getClickCount"] }),
};
</script>
  1. 組件觸發(fā)更新
<script>
export default {
  data() {
    return { times: 0 };
  },
  methods: {
    increment() {
      this.times++;
      //分發(fā)到action
      this.$store.dispatch("incrementAsync");
      //提交到mutations
      this.$store.commit("increment");
    },
  },
};
</script>

解析


Vuex 是什么鲫构?

Vuex 是一個(gè)專為 Vue.js 應(yīng)用程序開發(fā)的狀態(tài)管理模式。它采用集中式存儲(chǔ)管理應(yīng)用的所有組件的狀態(tài)玫坛,并以相應(yīng)的規(guī)則保證狀態(tài)以一種可預(yù)測的方式發(fā)生變化结笨。Vuex 也集成到 Vue 的官方調(diào)試工具devtools extension,提供了諸如零配置的 time-travel 調(diào)試湿镀、狀態(tài)快照導(dǎo)入導(dǎo)出等高級(jí)調(diào)試功能炕吸。

State - 數(shù)據(jù)源

Vuex 使用單一狀態(tài)樹——是的,用一個(gè)對(duì)象就包含了全部的應(yīng)用層級(jí)狀態(tài)勉痴。

Vue通過store選項(xiàng)赫模,調(diào)用Vue.use(Vuex)注入到每一個(gè)子組件中(類似路由)

組件獲取State

computed: {
    count () {
      return this.$store.state.count
 }
}

或者使用輔助函數(shù)mapState

computed: mapState({
    // 箭頭函數(shù)可使代碼更簡練
    count: state => state.count
})
Getter - 數(shù)據(jù)封裝讀取(類似屬性)

Getter 接受 state 作為其第一個(gè)參數(shù)

getters: {
    doneTodos: state => {
      return state.todos.filter(todo => todo.done)
    },
    getTodoById: (state) => (id) => {
      return state.todos.find(todo => todo.id === id)
    }
  }

通過屬性訪問

store.getters.doneTodos

通過方法訪問

store.getters.getTodoById(2)

Getters 也提供了一個(gè)輔助函數(shù)方便訪問(mapGetters )

Mutation - 進(jìn)行狀態(tài)更改的地方

定義Mutation

mutations: {
  increment (state, n) {
    state.count += n
  }
}

組件觸發(fā)變更

store.commit('increment', 1)

Mutations也提供輔助函數(shù)(mapMutations)

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')`
    })
  }
}

注意事項(xiàng)

  • Mutation 必須是同步函數(shù)
  • 最好提前在你的 store 中初始化好所有所需屬性蒸矛。
  • 需要在對(duì)象上添加新屬性時(shí)使用 Vue.set 或 替換舊對(duì)象
Action - 對(duì)Mutation封裝

Action 類似于 mutation瀑罗,不同在于:

  • Action 提交的是 mutation,而不是直接變更狀態(tài)雏掠。
  • Action 可以包含任意異步操作斩祭。

定義Action

actions: {
  increment ({ commit }) {
    commit('increment')
  }
}

組件分發(fā)Action

store.dispatch('increment')

支持異步方式分發(fā)

actions: {
        async incrementAsync({ commit }) {
            return new Promise((resolve, reject) => {
                setTimeout(() => {
                    try {
                        commit('increment');
                        resolve(new Date().getTime() + '  成功執(zhí)行');
                    } catch (e) {
                        reject(e);
                    }
                }, 1000)
            });
        }
    }

組件調(diào)用異步分發(fā)

this.$store.dispatch("incrementAsync").then(
        (data) => {
          console.log(data);
        },
        (err) => {
          console.log(err);
        }
      );

參考文章

Vuex

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市乡话,隨后出現(xiàn)的幾起案子停忿,更是在濱河造成了極大的恐慌,老刑警劉巖蚊伞,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件席赂,死亡現(xiàn)場離奇詭異,居然都是意外死亡时迫,警方通過查閱死者的電腦和手機(jī)颅停,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來掠拳,“玉大人癞揉,你說我怎么就攤上這事∧缗罚” “怎么了喊熟?”我有些...
    開封第一講書人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長姐刁。 經(jīng)常有香客問我芥牌,道長,這世上最難降的妖魔是什么聂使? 我笑而不...
    開封第一講書人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任壁拉,我火速辦了婚禮谬俄,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘弃理。我一直安慰自己溃论,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開白布痘昌。 她就那樣靜靜地躺著钥勋,像睡著了一般。 火紅的嫁衣襯著肌膚如雪辆苔。 梳的紋絲不亂的頭發(fā)上算灸,一...
    開封第一講書人閱讀 49,031評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音姑子,去河邊找鬼。 笑死测僵,一個(gè)胖子當(dāng)著我的面吹牛街佑,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播捍靠,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼沐旨,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了榨婆?” 一聲冷哼從身側(cè)響起磁携,我...
    開封第一講書人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎良风,沒想到半個(gè)月后谊迄,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡烟央,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年统诺,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片疑俭。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡粮呢,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出钞艇,到底是詐尸還是另有隱情啄寡,我是刑警寧澤,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布哩照,位于F島的核電站挺物,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏飘弧。R本人自食惡果不足惜姻乓,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一嵌溢、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蹋岩,春花似錦赖草、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至扣囊,卻和暖如春乎折,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背侵歇。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國打工骂澄, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人惕虑。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓坟冲,卻偏偏與公主長得像,于是被迫代替她去往敵國和親溃蔫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子健提,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 一、什么是Vuex Vuex 是一個(gè)專為 Vue.js 應(yīng)用程序開發(fā)的狀態(tài)管理模式伟叛。它采用集中式存儲(chǔ)管理應(yīng)用的所有...
    紫月凌楓閱讀 10,126評(píng)論 0 16
  • 一私痹、初識(shí)VueX 1.1 關(guān)于VueX VueX是適用于在Vue項(xiàng)目開發(fā)時(shí)使用的狀態(tài)管理工具。試想一下统刮,如果在一個(gè)...
    怪獸難吃素閱讀 325,148評(píng)論 24 314
  • 可用于登錄的狀態(tài)管理 一紊遵、vuex包含五個(gè)基本的對(duì)象: state:存儲(chǔ)狀態(tài),也就是變量侥蒙; getters:派生狀...
    林夕Yola閱讀 644評(píng)論 0 0
  • Vuex是什么癞蚕? Vuex 是一個(gè)專為 Vue.js應(yīng)用程序開發(fā)的狀態(tài)管理模式。它采用集中式存儲(chǔ)管理應(yīng)用的所有組件...
    蕭玄辭閱讀 3,106評(píng)論 0 6
  • Vuex 是一個(gè)專為 Vue.js 應(yīng)用程序開發(fā)的狀態(tài)管理模式辉哥。它采用集中式存儲(chǔ)管理應(yīng)用的所有組件的狀態(tài)桦山,并以相應(yīng)...
    尤雨溪的大迷弟閱讀 791評(píng)論 0 1