一. 概述
React
与Vue
是我们熟悉的两大前端主流框架,来自官方的解释,Vue是一套用于构建用户界面的渐进式框架,React是一个用于构建用户界面的JavaScript库,两个框架都使用各自的语法,专注于用户UI界面的构建.那我们会有疑问,这两个框架都专注于UI界面的构建,但是随着JavaScript单页应用开发日趋复杂,我们如何进行更多数据的管理呢?比如网络请求的数据、缓存数据、本地生成尚未持久化到服务器的数据,UI状态数据,激活的路由,被选中的标签等等. 基于上面的疑问,两个框架都有各自的解决方案:React-Redux
与Vuex
.
1. Redux 和 Vuex区别
先看Vuex作者尤雨溪的回答:
Vuex 其实是一个针对 Vue 特化的 Flux,主要是为了配合 Vue 本身的响应式机制。当然吸取了一些 Redux 的特点,比如单状态树和便于测试和热重载的 API,但是也选择性的放弃了一些在 Vue 的场景下并不契合的特性,比如强制的 immutability(在保证了每一次状态变化都能追踪的情况下强制的 immutability 带来的收益就很有限了)、为了同构而设计得较为繁琐的 API、必须依赖第三方库才能相对高效率地获得状态树的局部状态等等(相比之下 Vuex 直接用 Vue 本身的计算属性就可以)所以 Vue + Vuex 会更简洁,也不需要考虑性能问题,代价就是 Vuex 只能和 Vue 配合。Vue + Redux 也不是不可以,但是 Redux 作为一个泛用的实现和 Vue 的契合度肯定不如 Vuex。
Vuex改进了Redux中的Action和Reducer函数,以mutations变化函数取代Reducer,无需switch,只需在对应的mutation函数里改变state值即可
Vuex由于Vue自动重新渲染的特性,无需订阅重新渲染函数,只要生成新的State即可
Vuex数据流的顺序是∶View调用store.commit提交对应的请求到Store中对应的mutation函数->store改变(vue检测到数据变化自动渲染)
通俗点理解就是,vuex 弱化 dispatch,通过commit进行 store状态的一次更变;取消了action概念,不必传入特定的 action形式进行指定变更;弱化reducer,基于commit参数直接对数据进行转变,使得框架更加简易;
2. 共同思想
- 单—的数据源
- 变化可以预测
- 本质上∶ redux与vuex都是对mvvm思想的服务,将数据从视图中抽离的一种方案。
vuex 和 redux 都是状态管理库,用于单独管理状态的。其中,redux是一个范用的库,可以单独使用。而vuex是专门用来配合vue使用的。他们都应用了flux架构的思想,但是在接口的提供上稍有不同。
3.核心概念对比
Redux 的核心概念
- action (同步action ,或借助 中间件 实现异步操作,action 不会改变 store,只是描述了怎么改变store)| mutation(用于同步操作) 、action(可用于异步操作,提交 mutation)
- reducer(纯函数,根据 action 和旧的 store 计算出新的 store)
- store(单一数据源)
Vuex 的核心概念
- mutation(用于同步操作) 、action(可用于异步操作,提交 mutation)
- mutation里面直接修改 state
- state(单一数据源)
4. 使用原则:
Redux 的三大原则:
- 单一数据源(一个Redux应用只有一个store),也是单向的数据流
- state只读(唯一改变 state 的方法就是触发 action,action 是一个用于描述已发生事件的普通对象)
- 使用纯函数(reducer)来修改state。
redux的流程:
view——>action——>store——>reducer(返回)——>store——view
Vuex 的三大原则:
- 应用层级的状态应该集中到单个 store 对象中。
- 提交 mutation 是更改状态的唯一方法,并且这个过程是同步的。
- 异步逻辑都应该封装到 action 里面。
vue的流程:
vueComponent——>(dispatch)action——>(commit)——>mutations——>(mutate)state——>(render)vueComponent
5. 处理异步操作
Redux 的中间件机制,利用 redux-thunk ,redux-thunk可以dispatch函数,这个函数用于生成action,所以在这个函数里我们可以进行异步操作,等异步的结果出来后再放在action里面,将这个action用dispatch分发出去,而这个函数被叫做 “action creator”,可以将异步逻辑放在 action creator 里面,给 action creator 传入 dispatch 作为参数,于是就可以 dispatch action,Redux 并没有创造单独的概念出来专门用于异步逻辑,它是利用了 Redux 自己实现的中间件机制,中间件从 dispatch 一个异步 action 到 action 到达 reducer 之间处理 action,在这期间通过异步操作得到的结果可以放到 action 里面再通过 dispatch 分发到 reducer,以前 dispatch 一个 action 之后,这个 action 回立即到达 reducer ,所以是同步 action,现在在 action creator 里面,可以等待异步操作结果再生成 action 分发,所以叫做异步 action
而 Vuex 是用 mutation 来对应 Redux 的 action,另外 Vuex 又创造了一个 action 来提交 mutation 并通过异步提交 mutation 来实现异步操作结果能够修改 state.
二.使用
1.Redux
使用react-redux
之前我们先来了解一下Redux
。Redux
是 JavaScript 状态容器,提供可预测化的状态管理,Redux
由Flux
演变而来,当然除了和React
一起用外,还支持其它界面库,不过我们这里主要介绍它配合React进行使用.先来了解下它的几个核心概念:
(1) 核心概念
State
: 所谓的state
就是React
组件所依赖的状态对象。你可以在里面定义任何组件所依赖的状态。比如一个简单的todo
应用的state可能是这样
{
todos: [{
text: 'Eat food',
completed: true
}, {
text: 'Exercise',
completed: false
}],
visibilityFilter: 'SHOW_COMPLETED'
}
Action
:action
就是一个普通JavaScript对象,用来描述发生了什么.比如
{ type: 'ADD_TODO', text: 'Go to swimming pool' }
{ type: 'TOGGLE_TODO', index: 1 }
{ type: 'SET_VISIBILITY_FILTER', filter: 'SHOW_ALL' }
你可以把action
理解为一个描述发生了什么的指示器。在实际应用中,我们会dispatch(action)
,通过派发action来达到修改state的目的。这样做的好处是可以清晰地知道应用中到底发生了什么。
Reducer
:reducer
的作用是用来初始化整个Store
,并且串起state
与action
, 它是一个接收state
和action
并返回新state
的函数.我们可以通过区分不同的action类型,来处理并返回不同的state.
const StoreAction = (state = defaultState,action) => {
switch (action.type) {
case HOME_ACTION_UPDATE_STATE:
return {...state,...action.data};
case ADD_ARTICLELIST_STATE:
let newState = JSON.parse(JSON.stringify(state));/// 深拷贝
newState.articleList = newState.articleList.concat(action.data);
return newState;
default:
return state;
}
}
(2) 使用原则
使用Redux
进行数据管理时有三个原则需要注意
- 单一数据源
整个应用的state
被储存在一棵object tree
中,并且这个object tree
只存在于唯一一个store
中。 State
是只读的
唯一改变state
的方法就是触发action
,action
是一个用于描述已发生事件的普通对象。- 使用纯函数来执行修改
我们通过reducer接收先前的state
和action
,并返回新的state
,Reducer
必须是一个纯函数,所谓的纯函数就是一个函数的返回结果只依赖于它的参数,并且在执行过程中没有副作用。
(3)React Redux
react-redux
是Redux
官方提供的React
绑定库.他的使用也遵循上面的redux原则。
- 安装
npm install --save react-redux
-
流程
image.png
通过上面的流程图可以很清晰的明确Redux
的使用:
React组件
首先调用ActionCreators
里事先定义好的方法,得到一个actoion
,通过dispatch(action)
达到派发action
给Reducer
的目的。Reducer
通过接受的不同的action
来对state
数据进行处理,处理完成后,返回一个新的state
,state
变化后React组件
进行重新渲染。
-
使用
-
入口文件index.js
import React from 'react' import { render } from 'react-dom' import { Provider } from 'react-redux' import { createStore } from 'redux' import todoApp from './reducers' import App from './components/App' let store = createStore(todoApp) render( <Provider store={store}> <App /> </Provider>, document.getElementById('root') )
-
-
创建store/reducer.js
import { ADD_TODO_LIST_VALUE } from "./actionTypes"; /// 初始化数据 const defaultState = { todos: [{ text: 'Eat food', completed: true }, { text: 'Exercise', completed: false }], visibilityFilter: true, } /// Reducer 可以接受state,但是不能修改State ! export default (state = defaultState , action) => { switch (action.type) { case ADD_TODO_LIST_VALUE: const newState = JSON.parse(JSON.stringify(state));///将原来的state 做一次深拷贝 newState.todos.push(action.value); return newState; default: return state; } }
-
根据
reducer
创建store/index.js,import { createStore,compose,applyMiddleware } from 'redux'; import reducer from './reducer'; const store = createStore(reducer); export default store;
-
创建
actionCreators
store/actionCreators.jsimport { ADD_TODO_LIST_VALUE } from "./actionTypes" export const addItemListAction = (value) => ({ type:ADD_TODO_LIST_VALUE, value })
-
创建
actionTypes
专门用来存储action
的type值export const ADD_TODO_LIST_VALUE = 'add_todo_list_value';
-
React组件中使用
import React, { Component } from 'react'; import { addItemListAction } from '../pages/home/store/actionCreators'; import {connect} from 'react-redux'; class Customtodolist extends Component { render() { return ( <div className='todo-list' onClick={()=>{this.props.addListItem()}}> <ul> { this.props.todos.map((item,index) => <li key={index}>{index}:{item}</li> ) } </ul> </div> ); } } const mapStateToProps = (state) => { return { todos: state.todos } } const mapDispatchToProps = (dispatch) => { return { addListItem: () => { const item = {text: 'Eat food',completed: true} const actionCreator = addItemListAction(item); dispatch(actionCreator); } } } export default connect(mapStateToProps, mapDispatchToProps)(Customtodolist);
我们通过
react-redux
的connect
方法,将mapStateToProps与mapDispatchToProps 方法与组件链接,然后直接在类组件中通过this.props.XXX
的方式进行访问Store中的state. -
React hooks
中使用/// hooks 中使用react-redux import { useSelector, useDispatch } from 'react-redux'; const Home = (props)=> { // hook 获取store state 方式 const storeCount = useSelector(state => state.home.count); // 获取actionCreator方法 const dispatch = useDispatch(); return ( <div className={style['home-content']}> <div className='home-content-detail'> StoreCount数据展示{storeCount} <div> {/* addStoreCount是在actionCreators中定义的方法 */} <button onClick={()=> {dispatch(addStoreCount(0))}}>点击storeCount+1</button> </div> </div> </div> ) }
-
redux-thunk
的使用redux-thunk
是redux
的中间件.他的主要作用是可以使用异步派发action。例如我们要进行网络请求,那么可以在actionCreators
里面异步派发action.安装与使用
npm install --save redux-thunk
(1).在store/index.js 中添加引入
import thunk from "redux-thunk";/// redux-thunk 中间件 需要引入的
(2). 使用
thunk
初始化store
/// 下面的代码是固定写法 /// redux-thunk 中间件 需要引入的 const composeEnhancers = typeof window === 'object' && window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ ? window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__({ }) : compose; /// 中间件都需要用这个方法 const enhancer = composeEnhancers( applyMiddleware(thunk),/// redux-thunk 中间件 需要引入的 ); /// 创建 const store = createStore( reducer, enhancer//使用Redux-thunk 中间件 );
(3).在
actionCreater.js
中添加异步派发函数,注意:在获取到异步处理的结果后,我们仍然需要调用actionCreater.js
中的其他创建action方法,来对其进行dispatch
.export const initListAction = (data) => ({ type:INIT_LIST_ACTION, data }) /// 将异步请求 放在 Action 中执行 export const getToList = () => { /// 这里返回一个函数,能获取到 dispatch 方法 这就是 redux-thunk 的作用,可以返回一个函数 return (dispatch) => { axios.get('/api/todolist').then((res) => { // alert(res.data); const data = res.data; const action = initListAction(data); dispatch(action); }).catch((error)=>{ console.log('网络请求错误了---thunk----》'); }) } }
-
reducer
的拆分与合并随着项目功能模块越来越多,如果只有一个
reducer
来维护state
,会使其变动越来越大,从而导致难以维护。combineReducer
应运而生, 它将根reducer
分拆成多个reducer
,拆分之后的reducer
都是相同的结构(state, action),并且每个函数独立负责管理该特定切片 state 的更新。多个拆分之后的reducer
可以响应一个 action,在需要的情况下独立的更新他们自己的切片 state,最后组合成新的 state。使用
import { combineReducers } from 'redux';/// 将小的Reducer 合并成大的reducers /// 需要拆分 import headerReducer from '../common/header/store/reducer' import mainReducer from './mainReducer'; import {reducer as homeReducer} from '../pages/home/store'; import {reducer as loginReducer} from '../pages/login/store'; /// 进行 reducer的合并 const reducer = combineReducers({ header:headerReducer, main:mainReducer, login:loginReducer, home:homeReducer, }) export default reducer;
在
react
组件中使用,要加上reducer名称,例如我们在Home
组件中这样获取其stateconst mapStateToProps = (state, ownProps) => { return { showScroll: state.home.showScroll,//state后面添加reducer名称 } }
2.Vuex
Vuex
是一个专为 Vue.js 应用程序开发的状态管理模式 + 库。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。它主要用来解决多个组件共享状态的问题。
image.png
Vuex
跟 Redux
的数据管理模式很相似,如果理解Redux
,那么Vuex
也很容易理解了,只不过Vuex
是专门为 Vue.js 设计的状态管理库,使用起来要更加方便。
(1) 核心概念
-
State
: 就是组件所依赖的状态对象。我们可以在里面定义我们组件所依赖的数据。可以在Vue组件中通过this.$store.state.XXX
获取state里面的数据. -
Getter
:从store
中的state
中派生出一些状态,可以把他理解为是store
的计算属性.const store = createStore({ state: { todos: [ { id: 1, text: '...', done: true }, { id: 2, text: '...', done: false } ] }, getters: { doneTodos: (state) => { return state.todos.filter(todo => todo.done) } } })
例如我们定义了上面的store,在Vue组件中通过
store.getters.doneTodos
访问它的getter
. -
Mutation
:更改 Vuex 的 store 中状态的唯一方法是提交 mutation,我们通过在mutation中定义方法来改变state里面的数据。const store = createStore({ state: { count: 1 }, mutations: { increment (state) { // 变更状态 state.count++ } } })
在Vue组件中,我们通过
store.commit('increment')
,来提交。需要注意的是,Mutation 必须是同步函数。在实际使用中我们一般使用常量替代 Mutation 事件类型。例如:export const INCREMENT_MUTATION = 'INCREMENT_MUTATION' const store = createStore({ state: { count: 1 }, mutations: { // 我们可以使用 ES2015 风格的计算属性命名功能来使用一个常量作为函数名 [INCREMENT_MUTATION] (state) { // 变更状态 state.count++ } } })
-
Action:
Action 类似于 Mutation,不同在于:- Action 提交的是 mutation,而不是直接变更状态。
- Action 可以包含任意异步操作。
const store = createStore({ state: { count: 0 }, mutations: { increment (state) { state.count++ } }, actions: { incrementAsync ({ commit }) { setTimeout(() => { commit('increment') }, 1000) } } })
在组件中我们通过
store.dispatch('incrementAsync')
触发action。 -
Module
: 当我们的应用较大时,为了避免所有状态会集中到一个比较大的对象中,Vuex 允许我们将 store 分割成模块(module),你可以把它理解为Redux
中的combineReducer
的作用.const moduleA = { state: () => ({ ... }), mutations: { ... }, actions: { ... }, getters: { ... } } const moduleB = { state: () => ({ ... }), mutations: { ... }, actions: { ... } } const store = createStore({ modules: { a: moduleA, b: moduleB } })
在Vue组件中我们使用store.state.a, store.state.b
来分别获取两个模块的状态.
(2) 使用
-
安装
npm install vuex@next --save OR yarn add vuex@next --save
-
main.js
中挂载storeimport { createApp } from 'vue'; import App from './App.vue'; import router from './router'; import store from './store'; createApp(App).use(store).use(router).mount('#app');
-
创建
store/index.js
import { createStore } from 'vuex'; import { ADD_ITEM_LIST, REDUCE_ITEM_LIST, CHANGE_ITEM_LIST_ASYNC } from './constants'; export default createStore({ state: { itemList: [ { text: 'Learn JavaScript', done: true }, { text: 'Learn Vue', done: false }, { text: 'Build something awesome', done: false }, ], }, getters: { doneItemList: (state) => state.itemList.filter((todo) => todo.done), }, mutations: { // 使用ES2015风格的计算属性命名功能 来使用一个常量作为函数名 [ADD_ITEM_LIST](state, item) { console.log('增加数据', item); state.itemList.push(item); }, [REDUCE_ITEM_LIST](state) { console.log('减少数据'); state.itemList.pop(); }, }, actions: { [CHANGE_ITEM_LIST_ASYNC]({ commit, state }, todoItem) { /// 模拟网络请求 setTimeout(() => { commit(ADD_ITEM_LIST, todoItem); console.log('state===', state); }, 1000); }, }, modules: { }, });
注意我们这里仍然使用常量来作
actions
和mutations
的方法名,使用时候要加上[]
.
-
在选项式API中使用
<template> <div class="about"> <div>{{counter}}</div> <button @click="handleClick">按钮</button> <div> <div> <TodoItem :itemList="$store.state.itemList"/> </div> <div> 完成的Todo <TodoItem :itemList="$store.getters.doneItemList"/> </div> <div class="btn-content"> <button @click="addClick">增加Item</button> <button @click="reduceClick">减少Item</button> <button @click="changeClickAsync">调用Action</button> </div> </div> </div> </template> <script> import TodoItem from '../components/TodoItem.vue'; import { ADD_ITEM_LIST, REDUCE_ITEM_LIST, CHANGE_ITEM_LIST_ASYNC, } from '../store/constants'; export default { name: 'About', components: { TodoItem, }, data() { return { counter: 0, }; }, computed: { todos() { return this.$store.getters.doneItemList; }, }, methods: { handleClick() { console.log('handleClick--->'); this.counter += 1; }, addClick() { const item = { text: 'add_item_list success!', done: true }; /// 提交mutations this.$store.commit(ADD_ITEM_LIST, item); }, reduceClick() { this.$store.commit(REDUCE_ITEM_LIST); }, changeClickAsync() const item = { text: 'async_add_item_list success!', done: true }; ///派发actions this.$store.dispatch(CHANGE_ITEM_LIST_ASYNC, item); }, }, }; </script>
-
在组合式API中使用
在组合式API中通过调用
useStore
函数,来在setup
钩子函数中访问 store。这与在组件中使用选项式 API 访问this.$store
是等效的.import { useStore } from 'vuex' import { computed } from 'vue' export default { setup () { const store = useStore() return { // 在 computed 函数中访问 state count: computed(() => store.state.count), // 在 computed 函数中访问 getter double: computed(() => store.getters.double) // 使用 mutation increment: () => store.commit('increment'), // 使用 action asyncIncrement: () => store.dispatch('asyncIncrement') } } }