reactredux学习

54次阅读

共计 4986 个字符,预计需要花费 13 分钟才能阅读完成。

redux 学习笔记

学习动机

随着 JavaScript 单页应用开发日趋复杂,JavaScript 需要管理比任何时候都要多的 state(状态)。这些 state 可能包括服务器响应、缓存数据、本地生成尚未持久化到服务器的数据,也包括 UI 状态,如激活的路由,被选中的标签,是否显示加载动效或者分页器等等。

使用的情形

  • 用户的使用方式复杂
  • 不同身份的用户有不同的使用方式(比如普通用户和管理员)
  • 多个用户之间可以协作
  • 与服务器大量交互,或者使用了 WebSocket
  • View 要从多个来源获取数据
  • 从组件角度看,如果你的应用有以下场景,可以考虑使用 Redux。

    • 某个组件的状态,需要共享
    • 某个状态需要在任何地方都可以拿到
    • 一个组件需要改变全局状态
    • 一个组件需要改变另一个组件的状态

核心 api

createStore 可以帮助我们创建一个 store

store.dispatch 帮助我们派发 action 这个 action 会传递给 store

store.getState 获取到 store 里面所有的数据内容

store.subscribe 可以让我们订阅(监听) store 的改变 只要 store 发生改变,这个方法的回调函数就会执行。

Store

Store 就是保存数据的地方,你可以把它看成一个容器。整个应用只能有一个 Store。

Redux 提供 createStore 这个函数,用来生成 Store。

import {createStore} from 'redux';
const store = createStore(fn);

上面代码中,createStore函数接受另一个函数作为参数,返回新生成的 Store 对象。

State

Store对象包含所有数据。如果想得到某个时点的数据,就要对 Store 生成快照。这种时点的数据集合,就叫做 State。

当前时刻的 State,可以通过 store.getState() 拿到。

import {createStore} from 'redux';
const store = createStore(fn);

const state = store.getState();

Redux 规定,一个 State 对应一个 View。只要 State 相同,View 就相同。你知道 State,就知道 View 是什么样,反之亦然。

Action

State 的变化,会导致 View 的变化。但是,用户接触不到 State,只能接触到 View。所以,State 的变化必须是 View 导致的。Action 就是 View 发出的通知,表示 State 应该要发生变化了。

Action 是一个对象。其中的 type 属性是必须的,表示 Action 的名称。其他属性可以自由设置,社区有一个规范可以参考。

const action = {
type: 'ADD_TODO',
payload: 'Learn Redux'
};

上面代码中,Action 的名称是ADD_TODO,它携带的信息是字符串Learn Redux

可以这样理解,Action 描述当前发生的事情。改变 State 的唯一办法,就是使用 Action。它会运送数据到 Store。

Action Creator

View 要发送多少种消息,就会有多少种 Action。如果都手写,会很麻烦。可以定义一个函数来生成 Action,这个函数就叫 Action Creator。

const ADD_TODO = '添加 TODO';

function addTodo(text) {
return {
 type: ADD_TODO,
 text
}
}

const action = addTodo('Learn Redux');

上面代码中,addTodo函数就是一个 Action Creator。

store.dispatch()

store.dispatch()是 View 发出 Action 的唯一方法。

import {createStore} from 'redux';
const store = createStore(fn);

store.dispatch({
type: 'ADD_TODO',
payload: 'Learn Redux'
});

上面代码中,store.dispatch接受一个 Action 对象作为参数,将它发送出去。

结合 Action Creator,这段代码可以改写如下。

store.dispatch(addTodo('Learn Redux'));

Reducer

Store 收到 Action 以后,必须给出一个新的 State,这样 View 才会发生变化。这种 State 的计算过程就叫做 Reducer。

Reducer 是一个函数,它接受 Action 和当前 State 作为参数,返回一个新的 State。

const reducer = function (state, action) {
// ...
return new_state;
};

Reducer 函数最重要的特征是,它是一个纯函数。也就是说,只要是同样的输入,必定得到同样的输出。

纯函数是函数式编程的概念,必须遵守以下一些约束。

  • 不得改写参数
  • 不能调用系统 I/O 的 API
  • 不能调用 Date.now() 或者 Math.random() 等不纯的方法,因为每次会得到不一样的结果

由于 Reducer 是纯函数,就可以保证同样的 State,必定得到同样的 View。但也正因为这一点,Reducer 函数里面不能改变 State,必须返回一个全新的对象,请参考下面的写法。

// State 是一个对象
function reducer(state, action) {return Object.assign({}, state, {thingToChange});
// 或者
return {...state, ...newState};
}

// State 是一个数组
function reducer(state, action) {return [...state, newItem];
}

最好把 State 对象设成只读。你没法改变它,要得到新的 State,唯一办法就是生成一个新对象。这样的好处是,任何时候,与某个 View 对应的 State 总是一个不变的对象。

store.subscribe()

Store 允许使用 store.subscribe 方法设置监听函数,一旦 State 发生变化,就自动执行这个函数。

import {createStore} from 'redux';
const store = createStore(reducer);

store.subscribe(listener);

显然,只要把 View 的更新函数(对于 React 项目,就是组件的 render 方法或 setState 方法)放入listen,就会实现 View 的自动渲染。

redux 工作流程

首先,用户发出 Action。

store.dispatch(action);

然后,Store 自动调用 Reducer,并且传入两个参数:当前 State 和收到的 Action。Reducer 会返回新的 State。

let nextState = todoApp(previousState, action);

State 一旦有变化,Store 就会调用监听函数。

// 设置监听函数
store.subscribe(listener);

listener可以通过 store.getState() 得到当前状态。如果使用的是 React,这时可以触发重新渲染 View。

function listerner() {let newState = store.getState();
component.setState(newState);   
}

react-redux

Redux 官方提供的 React 绑定库。具有高效且灵活的特性。

安装

React Redux 依赖 React 0.14 或更新版本。

npm install --save react-redux
yarn add react-redux

API

<Provider store>

<Provider store> 使组件层级中的 connect() 方法都能够获得 Redux store。正常情况下,你的根组件应该嵌套在 <Provider> 中才能使用 connect() 方法。

// 自定义 App 组件  Provider 组件将 store 全部提供给被它包裹的组件。Provider 内部的组件都可以获得 store 里的数据。const App = (<Provider store={store}>
    <TodoList/>
  </Provider>
)

connect

connect([mapStateToProps], [mapDispatchToProps], [mergeProps], [options])

export default connect(mapStateToProps, mapDispatchToProps)(TodoList);  // connect 方法与 Provider 做连接  获取 store 里的数据

连接 React 组件与 Redux store。

连接操作不会改变原来的组件类。
反而 返回 一个新的已与 Redux store 连接的组件类。

参数

  • [mapStateToProps(state, [ownProps]): stateProps] (Function): 如果定义该参数,组件将会监听 Redux store 的变化。任何时候,只要 Redux store 发生改变,mapStateToProps 函数就会被调用。该回调函数必须返回一个纯对象,这个对象会与组件的 props 合并。如果你省略了这个参数,你的组件将不会监听 Redux store。如果指定了该回调函数中的第二个参数 ownProps,则该参数的值为传递到组件的 props,而且只要组件接收到新的 props,mapStateToProps 也会被调用(例如,当 props 接收到来自父组件一个小小的改动,那么你所使用的 ownProps 参数,mapStateToProps 都会被重新计算)。

mapStateToProps 函数的第一个参数是整个 Redux store 的 state,它返回一个要作为 props 传递的对象。

// 把 store 里的数据映射到组件的 props 里  
const mapStateToProps = (state) => {  //state 参数里的数据就是 store 内的数据
  return {
    inputValue: state.inputValue,
    list: state.list
  }
}
  • [mapDispatchToProps(dispatch, [ownProps]): dispatchProps] (Object or Function): 如果传递的是一个对象,那么每个定义在该对象的函数都将被当作 Redux action creator,对象所定义的方法名将作为属性名;每个方法将返回一个新的函数,函数中 dispatch 方法会将 action creator 的返回值作为参数执行。这些属性会被合并到组件的 props 中。

如果传递的是一个函数,该函数将接收一个 dispatch 函数,然后由你来决定如何返回一个对象,这个对象通过 dispatch 函数与 action creator 以某种方式绑定在一起

// store.dispatch, props 把 store.dispatch 方法映射到 props 里  通过 props 调用 dispatch 里的方法改变 store 的值
const mapDispatchToProps = (dispatch) => {
  return {changeInputValue(e) {
      const action = {
        type: 'change_input_value',
        value: e.target.value
      }
      dispatch(action)
    },
  }

简写语法注入 todos 和特定的 action 创建函数(addTodo and deleteTodo)

import {addTodo, deleteTodo} from './actionCreators'

function mapStateToProps(state) {return { todos: state.todos}
}

const mapDispatchToProps = {
  addTodo,
  deleteTodo
}

export default connect(
  mapStateToProps,
  mapDispatchToProps
)(TodoApp)

附上自己写的 react + redux 小 Demo 仓库:https://github.com/fuchengjx/…

参考博客

正文完
 0

react-redux-学习

54次阅读

共计 3642 个字符,预计需要花费 10 分钟才能阅读完成。

安装

npm install redux

核心概念

用户触发 action store 自动调用 reducer 更新 state

  1. store 用来存储整个应用的 state
  2. state 一个用来描述应用状态的对象
  3. action 一个对象用来描述程序发生的动作
  4. reducer 是更改 state 的纯函数接受 state 和 action 参数最后返回新的 state

下面我会对每个概念进行介绍

三大原则

  1. 单一的数据来源,整个应用程序的状态存储在单个 store 中
  2. state 对象是只读的改变 state 的唯一方式是发出一个 action
  3. 如何通过 action 改变 state,通过编写春函数 reducer 来实现

Actions

action是将数据发送的 store 的有效负载,他们是 store 数据的唯一来源通过 store.dispatch()将数据发送的 store

action 是一个普通的 js 对象,action 必须指定一个 type 属性用来描述执行的操作。

{
  type: 'ADD_TODO',
  text: 'Build my first Redux app'
}

Action Creators

action creators 是返回一个 action 的方法,这样写可以方便开发和测试。

function addTodo(text) {
  return {
    type: ADD_TODO,
    text
  }
}

dispatch(addTodo('eat'))

Reducers

Reducers 是指定状态树如何响应 dispatch 到 store 的 action 的,action 只是用来描述发生的动作,不描述状态的变化。

首先确定 state 的数据的格式

{
  todos: [
    {
      text: 'Consider using Redux',
      completed: true
    }
  ]
}

确定 state 的数据格式之后我们就可以编写一个 reducer,reducer 是一个纯函数接收 state 和 action 返回新的 state

const initialState = {todos: []
}

function todoApp(state = initialState, action) {switch (action.type) {
    case 'ADD_TODO':
      return Object.assign({}, state, {
        todos: [
          ...state.todos,
          {
            text: action.text,
            completed: false
          }
        ]
      })
    default:
      return state
  }
}

注意我们使用 Object.assign()返回一个新的状态,不改变参数中的 state 的值

分割 Reducers

当我们应用程序的 action 比较多每个 action 在 reducers 中的逻辑比较复杂的时候,如果还像我们上面一直使用 switch case 可以想象我们的 reducers 会变的非常的长,此时我们就需要对一些复杂的操作进行分割,分割的规则是根据 state 的节点操作进行分割,针对每个节点的操作写一个 reducers 然后使用 combineReducers 方法进行合并。

例如

import {combineReducers} from 'redux'
import {
  ADD_TODO,
  TOGGLE_TODO,
  SET_VISIBILITY_FILTER,
  VisibilityFilters
} from './actions'
const {SHOW_ALL} = VisibilityFilters

function visibilityFilter(state = SHOW_ALL, action) {switch (action.type) {
    case SET_VISIBILITY_FILTER:
      return action.filter
    default:
      return state
  }
}

function todos(state = [], action) {switch (action.type) {
    case ADD_TODO:
      return [
        ...state,
        {
          text: action.text,
          completed: false
        }
      ]
    default:
      return state
  }
}

const todoApp = combineReducers({
  visibilityFilter,
  todos
})

export default todoApp

Store

在 redux 应用程序中只能有一个 store,使用 store 可以将上面的 reducers 和 actions 进行组合,然后进行操作。

创建一个 store

redux 提供了一个 createStore 接收一个 reducers 参数,最终返回一个 store,我们将之前定义好的 reducers 引入,然后传入 createStore 就可以了。

import {createStore} from 'redux'
import todoApp from './reducers'
const store = createStore(todoApp)

store 提供的方法

  1. getState() 获取 store 中的状态树
  2. dispatch(action) 发送 action 更新 action
  3. subscribe(listener) 订阅监听返回一个 unsubscribe 函数用来取消订阅

react-redux

安装

npm install --save react-redux

Provider

<provider /> 是 redux 提供的一个组件,这个组件 有一个 props store (上面文档中创建出来的 store) , 这个组件可以使,使用 connect()函数嵌套的组件能访问 redux 中的 store。

用法

import React from 'react'
import ReactDOM from 'react-dom'
import {Provider} from 'react-redux'

import {App} from './App'
import createStore from './createReduxStore'

const store = createStore()

ReactDOM.render(<Provider store={store}>
    <App />
  </Provider>,
  document.getElementById('root')
)

connect

connect函数用于把 redux 的 store 和 react 的组件连接到一起,他返回的是一个包装函数,这个函数的参数是一个 react 的组件,这个函数将会返回一个 包装组件。

function connect(mapStateToProps?, mapDispatchToProps?, mergeProps?, options?)

例子

import {addTodo} from './actionCreators'

const mapDispatchToProps = {addTodo}
const mapStateToProps = (state, ownProps) => ({todo: state.todos[ownProps.id]
})
export default connect(
  mapStateToProps,
  mapDispatchToProps
)(TodoApp)

mapStateToProps?: (state, ownProps?) => Object

state 为 redux 的状态树,ownProps 是链接组件的 props

如果此方法有一个参数,在 redux 的 state 发生变化的时候会调用函数,如果此方法有两个参数,此方法将会在连接组件的 props 或者 redux 的 state 发生变化的时候调用。

此方法返回一个对象, 此对象(通常称为 stateProps)将合并为连接组件的props 。如果定义 mergeProps,它将作为 mergeProps 的第一个参数提供。

mapDispatchToProps?: Object | (dispatch, ownProps?) => Object

dispatch 即 store 中的 dispatch 通过传入一个 action 给变 store 中的 state 数据

ownProps 即 链接组件的 props

您的 mapDispatchToProps 函数应返回一个对象。对象的每个字段都应该是一个函数,调用哪个函数可以 dispatch 一个 action 到 store。

mapDispatchToProps 函数的返回被视为 dispatchProps。它将作为 props 合并到您连接的组件。如果定义 mergeProps,它将作为 mergeProps 的第二个参数提供。

mergeProps?: (stateProps, dispatchProps, ownProps) => Object

如果指定,则定义如何确定自己的包装组件的最终 props。如果您不提供 mergeProps,则默认情况下,您的包装组件会收到{… ownProps,… stateProps,… dispatchProps}。

mergeProps 的返回值称为 mergedProps,字段将用作包装组件的 props。

options?: Object

暂未使用

正文完
 0