- ReactJS 教程
- ReactJS - 首页
- ReactJS - 简介
- ReactJS - 路线图
- ReactJS - 安装
- ReactJS - 特性
- ReactJS - 优点与缺点
- ReactJS - 架构
- ReactJS - 创建 React 应用
- ReactJS - JSX
- ReactJS - 组件
- ReactJS - 嵌套组件
- ReactJS - 使用新创建的组件
- ReactJS - 组件集合
- ReactJS - 样式
- ReactJS - 属性 (props)
- ReactJS - 使用属性创建组件
- ReactJS - props 验证
- ReactJS - 构造函数
- ReactJS - 组件生命周期
- ReactJS - 事件管理
- ReactJS - 创建一个事件感知组件
- ReactJS - 在 Expense Manager 应用中引入事件
- ReactJS - 状态管理
- ReactJS - 状态管理 API
- ReactJS - 无状态组件
- ReactJS - 使用 React Hooks 进行状态管理
- ReactJS - 使用 React Hooks 进行组件生命周期管理
- ReactJS - 布局组件
- ReactJS - 分页
- ReactJS - Material UI
- ReactJS - Http 客户端编程
- ReactJS - 表单编程
- ReactJS - 受控组件
- ReactJS - 非受控组件
- ReactJS - Formik
- ReactJS - 条件渲染
- ReactJS - 列表
- ReactJS - Keys
- ReactJS - 路由
- ReactJS - Redux
- ReactJS - 动画
- ReactJS - Bootstrap
- ReactJS - 地图
- ReactJS - 表格
- ReactJS - 使用 Flux 管理状态
- ReactJS - 测试
- ReactJS - CLI 命令
- ReactJS - 构建与部署
- ReactJS - 示例
- Hooks
- ReactJS - Hooks 简介
- ReactJS - 使用 useState
- ReactJS - 使用 useEffect
- ReactJS - 使用 useContext
- ReactJS - 使用 useRef
- ReactJS - 使用 useReducer
- ReactJS - 使用 useCallback
- ReactJS - 使用 useMemo
- ReactJS - 自定义 Hooks
- ReactJS 高级
- ReactJS - 可访问性
- ReactJS - 代码分割
- ReactJS - 上下文
- ReactJS - 错误边界
- ReactJS - 转发 Refs
- ReactJS - 片段
- ReactJS - 高阶组件
- ReactJS - 集成其他库
- ReactJS - 优化性能
- ReactJS - Profiler API
- ReactJS - 端口
- ReactJS - 无 ES6 ECMAScript 的 React
- ReactJS - 无 JSX 的 React
- ReactJS - 调和
- ReactJS - Refs 和 DOM
- ReactJS - 渲染 Props
- ReactJS - 静态类型检查
- ReactJS - 严格模式
- ReactJS - Web Components
- 其他概念
- ReactJS - 日期选择器
- ReactJS - Helmet
- ReactJS - 内联样式
- ReactJS - PropTypes
- ReactJS - BrowserRouter
- ReactJS - DOM
- ReactJS - 走马灯
- ReactJS - 图标
- ReactJS - 表单组件
- ReactJS - 参考 API
- ReactJS 有用资源
- ReactJS - 快速指南
- ReactJS - 有用资源
- ReactJS - 讨论
ReactJS - 使用 useContext
Context 是 React 中一个重要的概念。它提供了一种能力,可以将信息从父组件传递给其所有子组件,直到任何嵌套级别,而无需在每个级别通过 props 传递信息。Context 将使代码更具可读性和易于理解。Context 可用于存储不更改或更改最少的信息。Context 的一些用例如下:
应用程序配置
当前已认证用户信息
当前用户设置
语言设置
应用程序/用户主题/设计配置
React 提供了一个特殊的 hook,useContext,用于在函数组件中访问和更新上下文信息。让我们在本节中学习上下文及其相应的 hook。
Context 如何工作?
在了解 useContext hook 之前,让我们回顾一下 Context 的基本概念以及它的工作原理。Context 有四个部分,
创建一个新的上下文
在根组件中设置上下文提供者
在我们需要上下文信息的组件中设置上下文消费者
访问上下文信息并在渲染方法中使用它
让我们创建一个应用程序来更好地理解 Context 及其用法。让我们为在应用程序根组件中维护主题信息创建一个全局上下文,并在我们的子组件中使用它。
首先,使用以下命令创建并启动一个应用程序:
create-react-app myapp cd myapp npm start
接下来,在 components 文件夹下创建一个组件 HelloWorld(src/components/HelloWorld.js)
import React from "react"; import ThemeContext from "../ThemeContext"; class HelloWorld extends React.Component { render() { return <div>Hello World</div> } } export default HelloWorld
接下来,创建一个新的上下文 (src/ThemeContext.js) 来维护主题信息。
import React from 'react' const ThemeContext = React.createContext({ color: 'black', backgroundColor: 'white' }) export default ThemeContext
这里,
使用 React.createContext 创建了一个新的上下文。
上下文被建模为一个包含样式信息的的对象。
设置文本颜色和背景的初始值。
接下来,通过包含 HelloWorld 组件和带有主题上下文初始值的主题提供程序来更新根组件 App.js。
import './App.css'; import HelloWorld from './components/HelloWorld'; import ThemeContext from './ThemeContext' function App() { return ( <ThemeContext.Provider value={{ color: 'white', backgroundColor: 'green' }}> <HelloWorld /> </ThemeContext.Provider> ); } export default App;
这里,使用了 ThemeContext.Provider,它是一个非视觉组件,用于设置要在其所有子组件中使用的主题上下文的值。
接下来,在 HelloWorld 组件中包含一个上下文消费者,并使用 HelloWorld 组件中的主题信息来设置 hello world 消息的样式。
import React from "react"; import ThemeContext from "../ThemeContext"; class HelloWorld extends React.Component { render() { return ( <ThemeContext.Consumer> { ( {color, backgroundColor} ) => (<div style={{ color: color, backgroundColor: backgroundColor }}> Hello World </div>) } </ThemeContext.Consumer>) } } export default HelloWorld
这里我们有,
使用了 ThemeContext.Consumer,它是一个非视觉组件,提供对当前主题上下文详细信息的访问
使用函数表达式在 ThemeContext.Consumer 内部获取当前上下文信息
使用对象解构语法获取主题信息并将值设置为 color 和 backgroundColor 变量。
使用主题信息通过 style props 设置组件的样式。
最后,打开浏览器并检查应用程序的输出
useContext 的签名
useContext 的签名如下:
let contextValue = useContext( <contextName> )
这里,
contextName 指的是要访问的上下文的名称。
contextValue 指的是所引用上下文的当前值。
使用 hook 访问上下文的示例代码如下:
const theme = useContext(ThemContext)
通过 hook 使用 Context
让我们更新我们的应用程序并使用上下文 hook 而不是上下文消费者。
首先,将 HelloWorld 组件转换为函数组件。
import React from "react"; function HelloWorld() { return <div>Hello World</div> } export default HelloWorld
接下来,通过 useContext hook 访问上下文的当前值
import React, { useContext } from "react" import ThemeContext from '../ThemeContext' function HelloWorld() { let theme = useContext(ThemeContext) return <div>Hello World</div> } export default HelloWorld
接下来,更新渲染函数以使用通过上下文获取的主题信息。
import React, { useContext } from "react" import ThemeContext from '../ThemeContext' function HelloWorld() { let theme = useContext(ThemeContext) return ( <div style={{ color: theme.color, backgroundColor: theme.backgroundColor }}> Hello World </div> ) } export default HelloWorld
这里我们有,
使用 useContext 访问 ThemeContext 上下文信息。
使用 ThemeContext 信息设置文本的背景颜色和颜色。
最后,打开浏览器并检查应用程序的输出。
更新 Context
在某些情况下,更新上下文信息是必要的。例如,我们可以提供一个选项让用户更改主题信息。当用户更改主题时,上下文应该更新。更新上下文将重新渲染所有子组件,这将更改应用程序的主题。
React 提供了一个选项,可以使用 useState 和 useContext hook 来更新上下文。让我们更新我们的应用程序以支持主题选择。
首先,更新根组件 App.js 并使用 useState hook 来管理主题信息,如下所示:
import './App.css' import { useState } from 'react' import HelloWorld from './components/HelloWorld' import ThemeContext from './ThemeContext' function App() { let initialTheme = { color: 'white', backgroundColor: 'green' } const [theme, setTheme] = useState(initialTheme) return ( <ThemeContext.Provider value={{ theme, setTheme }}> <HelloWorld /> </ThemeContext.Provider> ); } export default App;
这里我们有,
使用 useState hook 在根组件的状态中设置主题信息。
主题更新函数 useTheme 也作为主题信息的一部分包含在上下文中。
接下来,更新 HelloWorld 组件以获取存储在上下文中的主题信息。
import React, { useContext } from "react" import ThemeContext from '../ThemeContext' function HelloWorld() { let { theme, setTheme } = useContext(ThemeContext) return (<div style={{ color: theme.color, backgroundColor: theme.backgroundColor }}> <div>Hello World</div> </div>) } export default HelloWorld
接下来,为用户提供一个通过下拉选项更改主题的选项。
import React, { useContext } from "react" import ThemeContext from '../ThemeContext' function HelloWorld() { let { theme, setTheme } = useContext(ThemeContext) return (<div style={{ color: theme.color, backgroundColor: theme.backgroundColor }}> <div> <select value={theme.backgroundColor}> <option value="green">Green</option> <option value="red">Red</option> </select> <div>Hello World</div> </div>) } export default HelloWorld
这里我们有,
添加了一个带有两个选项(绿色和红色)的下拉框。
使用当前主题值 value={theme.backgroundColor) 设置下拉框的当前值。
接下来,每当用户通过 onChange 事件更改主题时,更新上下文。
import React, { useContext } from "react" import ThemeContext from '../ThemeContext' function HelloWorld() { let { theme, setTheme } = useContext(ThemeContext) return (<div style={{ color: theme.color, backgroundColor: theme.backgroundColor }}> <div> <select value={theme.backgroundColor} onChange = { (e) => { setTheme({ ...theme, backgroundColor: e.target.value }) }} > <option value="green">Green</option> <option value="red">Red</option> </select> </div> <div>Hello World</div> </div>) } export default HelloWorld
这里我们有,
将 onChange 事件附加到下拉框。
在事件处理程序内部使用 setTheme 函数并将主题的背景颜色更新为用户选择的颜色。
根组件和 HelloWorld 组件的完整代码如下:
import React, { useContext } from "react" import ThemeContext from '../ThemeContext' function HelloWorld() { let { theme, setTheme } = useContext(ThemeContext) return (<div style={{ color: theme.color, backgroundColor: theme.backgroundColor }}> <div> <select value={theme.backgroundColor} onChange= { (e) => { setTheme({ ...theme, backgroundColor: e.target.value }) } } > <option value="green">Green</option> <option value="red">Red</option> </select> </div> <div>Hello World</div> </div>) } export default HelloWorld
接下来,打开浏览器并检查应用程序。
当用户选择不同的背景颜色时,它将更新上下文,并因此重新渲染组件,并使用新的主题,如下所示:
总结
Context 降低了在 React 应用程序中维护全局数据的复杂性。Context hook 通过简化访问和更新(通过 useState)上下文进一步降低了复杂性。