Redux参考手册
基础 | Basics
React用法(Usage with React)
从一开始,我们需要强调 Redux 与 React 没有任何关系。您可以使用 React,Angular,Ember,jQuery 或 vanilla JavaScript 编写 Redux 应用程序。
也就是说,Redux与React和Deku等库合作得非常好,因为它们让您将UI描述为状态的函数,并且Redux响应操作发出状态更新。
我们将使用React来构建我们简单的待办事项应用程序。
安装React Redux
React 绑定不包含在默认的 Redux 中。你需要明确地安装它们:
npm install --save react-redux如果您不使用 npm,您可以从 unpkg 获取最新的 UMD 版本(开发版或生产版)。window.ReactRedux如果您通过<script>标记将其添加到您的网页,则 UMD 版本将导出一个全局调用。
展示和容器组件
Redux 的 React 绑定包含分离表示和容器组件的想法。如果您不熟悉这些条款,请先阅读有关条款,然后再回来。这些条款很重要,所以我们会等待!
完成阅读文章?我们来重述一下他们的区别:
|  | 演示组件 | 容器组件 | 
|---|---|---|
| 目标 | How things look (markup, styles) | How things work (data fetching, state updates) | 
| Aware of Redux | No | Yes | 
| 读取数据 | Read data from props | Subscribe to Redux state | 
| 改变数据 | Invoke callbacks from props | Dispatch Redux actions | 
| 写入 | By hand | Usually generated by React Redux | 
我们要编写的大多数组件都是表示性的,但我们需要生成一些容器组件以将它们连接到 Redux 存储。这和下面的设计概要并不意味着容器组件必须靠近组件树的顶部。如果一个容器组件变得太复杂了(即它有大量嵌套的 presentional 组件,并有无数的回调被传递下去),请在 FAQ 中引入组件树中的另一个容器。
技术上你可以手动使用store.subscribe()容器组件。我们不建议您这样做,因为 React Redux 会进行许多难以完成的性能优化。出于这个原因,我们将使用connect()React Redux提供的函数生成它们,而不是编写容器组件,如下所示。
设计组件层次结构
请记住我们如何设计根状态对象的形状?是时候我们设计 UI 层次结构来匹配它。这不是特定于 Redux 的任务。在 React 中思考是一个很好的教程,可以解释这个过程。
我们的设计简介很简单。我们想要显示待办事项列表。单击时,待办事项完成后划掉。我们想要显示一个字段,用户可以添加新的待办事项。在页脚中,我们想要显示切换显示全部,只显示完成或仅显示活动待办事项。
设计演示组件
我看到以下演示组件和它们的道具出现在这个简短的介绍中:
- 
TodoList是一个列表,显示可见的待办事项。- 
todos: Array是具有{ id, text, completed }形状的待办事项的数组。
- 
onTodoClick(id: number)是单击todo时调用的回调函数。
 
- 
- 
Todo是一个单一的待办事项。- 
text: string是要显示的文字。
- 
completed: boolean是待办事项是否应该出现划掉。
- 
onClick()是单击todo时调用的回调函数。
 
- 
- 
Link是一个回调链接。- 
onClick()是单击链接时调用的回调。
 
- 
- 
Footer是我们让用户更改当前可见的待办事项的地方。
- 
App是呈现其他所有内容的根组件。
他们描述的外观,但不知道数据来自哪里,或者如何改变它。他们只渲染给他们的东西。如果你从 Redux 迁移到其他的东西,你将能够保持所有这些组件完全一样。他们没有依赖 Redux 。
设计容器组件
我们还需要一些容器组件将演示组件连接到 Redux。例如,演示TodoList组件需要一个像VisibleTodoList这样的容器订阅Redux商店,并知道如何应用当前的可见性过滤器。要更改可见性过滤器,我们将提供一个FilterLink容器组件,呈现一个Link在点击时发送适当操作的容器组件:
- 
VisibleTodoList根据当前可见性过滤器过滤待办事项并呈现TodoList。
- 
FilterLink获取当前可见性过滤器并呈现一个Link。- 
filter: string是它代表的可见性过滤器。
 
- 
设计其他组件
有时很难判断某个组件应该是一个表示组件还是一个容器。例如,有时窗体和函数真的耦合在一起,比如这个微小的组件:
- 
AddTodo是一个带有“添加”按钮的输入字段
从技术上讲,我们可以将它分成两个部分,但现阶段可能为时过早。在非常小的组件中混合呈现和逻辑是很好的。随着它的增长,如何分割它将会更加明显,所以我们会把它混合起来。
实现组件
我们来编写组件!我们从演示组件开始,所以我们不需要考虑绑定到 Redux。
演示组件
这些都是普通的 React 组件,所以我们不会详细检查。我们编写功能无状态的组件,除非我们需要使用本地状态或生命周期方法。这并不意味着表示组件必须是功能 - 这样更容易定义它们。如果您需要添加本地状态,生命周期方法或性能优化,则可以将它们转换为类。
components/Todo.js
import React from 'react'
import PropTypes from 'prop-types'
const Todo = ({ onClick, completed, text }) => (
  <li
    onClick={onClick}
    style={{
      textDecoration: completed ? 'line-through' : 'none'
    }}
  >
    {text}
  </li>
)
Todo.propTypes = {
  onClick: PropTypes.func.isRequired,
  completed: PropTypes.bool.isRequired,
  text: PropTypes.string.isRequired
}
export default Todocomponents/TodoList.js
import React from 'react'
import PropTypes from 'prop-types'
import Todo from './Todo'
const TodoList = ({ todos, onTodoClick }) => (
  <ul>
    {todos.map(todo => (
      <Todo key={todo.id} {...todo} onClick={() => onTodoClick(todo.id)} />
    ))}
  </ul>
)
TodoList.propTypes = {
  todos: PropTypes.arrayOf(
    PropTypes.shape({
      id: PropTypes.number.isRequired,
      completed: PropTypes.bool.isRequired,
      text: PropTypes.string.isRequired
    }).isRequired
  ).isRequired,
  onTodoClick: PropTypes.func.isRequired
}
export default TodoListcomponents/Link.js
import React from 'react'
import PropTypes from 'prop-types'
const Link = ({ active, children, onClick }) => {
  if (active) {
    return <span>{children}</span>
  }
  return (
    <a
      href="#"
      onClick={e => {
        e.preventDefault()
        onClick()
      }}
    >
      {children}
    </a>
  )
}
Link.propTypes = {
  active: PropTypes.bool.isRequired,
  children: PropTypes.node.isRequired,
  onClick: PropTypes.func.isRequired
}
export default Linkcomponents/Footer.js
import React from 'react'
import FilterLink from '../containers/FilterLink'
const Footer = () => (
  <p>
    Show:
    {' '}
    <FilterLink filter="SHOW_ALL">
      All
    </FilterLink>
    {', '}
    <FilterLink filter="SHOW_ACTIVE">
      Active
    </FilterLink>
    {', '}
    <FilterLink filter="SHOW_COMPLETED">
      Completed
    </FilterLink>
  </p>
)
export default Footercomponents/App.js
import React from 'react'
import Footer from './Footer'
import AddTodo from '../containers/AddTodo'
import VisibleTodoList from '../containers/VisibleTodoList'
const App = () => (
  <div>
    <AddTodo />
    <VisibleTodoList />
    <Footer />
  </div>
)
export default App实现容器组件
现在是时候通过创建一些容器来将这些表示组件连接到 Redux。从技术上讲,容器组件只是一个React组件,用于store.subscribe()读取Redux状态树的一部分,并为它呈现的呈现组件提供道具。您可以手动编写容器组件,但我们建议使用 React Redux 库的connect()函数生成容器组件,该函数提供了许多有用的优化以防止不必要的重新渲染。(这样做的一个结果就是你不必担心自己实现React性能的建议shouldComponentUpdate。)
要使用connect(),您需要定义一个特殊的函数调用mapStateToProps,告诉如何将当前的 Redux 存储状态转换为要传递给要包装的表示组件的道具。例如,VisibleTodoList需要计算todos传递给TodoList,所以我们根据过滤的函数state.visibilityFilter定义一个state.todos,并在其中使用它mapStateToProps:
const getVisibleTodos = (todos, filter) => {
  switch (filter) {
    case 'SHOW_ALL':
      return todos
    case 'SHOW_COMPLETED':
      return todos.filter(t => t.completed)
    case 'SHOW_ACTIVE':
      return todos.filter(t => !t.completed)
  }
}
const mapStateToProps = state => {
  return {
    todos: getVisibleTodos(state.todos, state.visibilityFilter)
  }
}除了读取状态之外,容器组件还可以调度操作。以类似的方式,您可以定义一个名为mapDispatchToProps()的函数来接收dispatch()方法并返回要注入到表示组件中的回调支持。例如,我们要VisibleTodoList注入一种叫做道具onTodoClick到TodoList组件,我们希望onTodoClick分派一个TOGGLE_TODO动作:
const mapDispatchToProps = dispatch => {
  return {
    onTodoClick: id => {
      dispatch(toggleTodo(id))
    }
  }
}最后,我们通过调用connect()并传递这两个函数来创建VisibleTodoList:
import { connect } from 'react-redux'
const VisibleTodoList = connect(
  mapStateToProps,
  mapDispatchToProps
)(TodoList)
export default VisibleTodoList这些是 React Redux API 的基础知识,但有几个捷径和电源选项,因此我们鼓励您详细查看其文档。如果您担心mapStateToProps经常创建新对象,则可能需要了解重新选择计算派生数据。
查找下面定义的其余容器组件:
containers/FilterLink.js
import { connect } from 'react-redux'
import { setVisibilityFilter } from '../actions'
import Link from '../components/Link'
const mapStateToProps = (state, ownProps) => {
  return {
    active: ownProps.filter === state.visibilityFilter
  }
}
const mapDispatchToProps = (dispatch, ownProps) => {
  return {
    onClick: () => {
      dispatch(setVisibilityFilter(ownProps.filter))
    }
  }
}
const FilterLink = connect(
  mapStateToProps,
  mapDispatchToProps
)(Link)
export default FilterLinkcontainers/VisibleTodoList.js
import { connect } from 'react-redux'
import { toggleTodo } from '../actions'
import TodoList from '../components/TodoList'
const getVisibleTodos = (todos, filter) => {
  switch (filter) {
    case 'SHOW_ALL':
      return todos
    case 'SHOW_COMPLETED':
      return todos.filter(t => t.completed)
    case 'SHOW_ACTIVE':
      return todos.filter(t => !t.completed)
  }
}
const mapStateToProps = state => {
  return {
    todos: getVisibleTodos(state.todos, state.visibilityFilter)
  }
}
const mapDispatchToProps = dispatch => {
  return {
    onTodoClick: id => {
      dispatch(toggleTodo(id))
    }
  }
}
const VisibleTodoList = connect(
  mapStateToProps,
  mapDispatchToProps
)(TodoList)
export default VisibleTodoList实施其他组件
containers/AddTodo.js
import React from 'react'
import { connect } from 'react-redux'
import { addTodo } from '../actions'
let AddTodo = ({ dispatch }) => {
  let input
  return (
    <div>
      <form
        onSubmit={e => {
          e.preventDefault()
          if (!input.value.trim()) {
            return
          }
          dispatch(addTodo(input.value))
          input.value = ''
        }}
      >
        <input
          ref={node => {
            input = node
          }}
        />
        <button type="submit">
          Add Todo
        </button>
      </form>
    </div>
  )
}
AddTodo = connect()(AddTodo)
export default AddTodo通过存储
所有容器组件都需要访问 Redux 存储,以便他们可以订阅。一种选择是将其作为道具传递给每个容器组件。然而,它很乏味,因为store即使通过演示组件来渲染容器深处的组件,也必须连线。
我们推荐的选择是使用所谓的特殊阵营终极版组件<Provider>神奇地使存储提供应用程序中的所有容器组件没有传递给它明确。渲染根组件时,只需要使用它一次:
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')
)下一步
阅读本教程的完整源代码以更好地内化您获得的知识。然后,请直接前往高级教程以了解如何处理网络请求和路由!
基础 | Basics相关
 
                                Redux由Dan Abramov在2015年创建的科技术语。是受2014年Facebook的Flux架构以及函数式编程语言Elm启发。很快,Redux因其简单易学体积小在短时间内成为最热门的前端架构。
| 主页 | http://redux.js.org/ | 
| 源码 | https://github.com/reactjs/redux/ | 
| 发布版本 | 3.7.2 | 
 
         加载中,请稍侯......
 加载中,请稍侯......