React用法(Usage with 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
是一个列表,显示可见的待办事项。
他们描述的外观,
但不知道数据来自哪里
,或者如何
改变它。他们只渲染给他们的东西。如果你从 Redux 迁移到其他的东西,你将能够保持所有这些组件完全一样。他们没有依赖 Redux 。
设计容器组件
我们还需要一些容器组件将演示组件连接到 Redux。例如,演示TodoList
组件需要一个像VisibleTodoList
这样的容器订阅Redux商店,并知道如何应用当前的可见性过滤器。要更改可见性过滤器,我们将提供一个FilterLink
容器组件,呈现一个Link
在点击时发送适当操作的容器组件:
VisibleTodoList
根据当前可见性过滤器过滤待办事项并呈现TodoList
。
设计其他组件
有时很难判断某个组件应该是一个表示组件还是一个容器。例如,有时窗体和函数真的耦合在一起,比如这个微小的组件:
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 Todo
components/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 TodoList
components/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 Link
components/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 Footer
components/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 FilterLink
containers/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')
)
下一步
阅读本教程的完整源代码以更好地内化您获得的知识。然后,请直接前往高级教程以了解如何处理网络请求和路由!