使用React开发应用程序时,如果多个组件需要共享数据,可以把数据放到父组件中,通过属性向下传递给子组件。但当组件嵌套较深时,两个最底层的组件要想共享数据,那就霜要把数据放到最顶层的组件中,然后再一层一层向下传递,可能需要向下传递好多层才能到达想要数据的子组件,这就产生了一个问题,由于经过的这些层(组件)可能不需要这个数据,在向下传递的过程中,有可能就忘记写共享属性,程序也就出错了,并且还不好调试。有没有一种方法,可以穿透组件,想要数据的子组件直接获取到最顶层组件的共享数据, 这就是React context。

  Context就是上下文,环境的意思,React 的context 就是提供了一个上下文或环境,在这个环境中,有context提供者和context消费者,提供者(provider) 提供共享数据,消费者(consumer) 消费这些共享数据。怎么提供和消费呢?provider和consumer是两个组件,provider有一个value属性, 只要把共享的数据放到value中,然后再把consumer组件包起来,consumer组件就能获到到共享数据。consumer组件消费数据的方式有两种,传统的是组件内部是回调函数,现代一点的是useContext hooks。

  create-react-app react-context 创建项目 ,cd react-context && npm install bootstrap。j假设Header中有一个user下拉框,选择user,然后Body中显示user,胡乱编造的一个例子。

  App.js

import React from "react";
import 'bootstrap/dist/css/bootstrap.css'; // 添加bootstrap 样式 import Header from "./Header";
import Details from './Details'; export default function App() {
return (
<div className="card">
<div className="card-header">
<Header></Header>
</div>
<div className="card-body">
<Details></Details>
</div>
</div>
)
}

  Header.js  

import React from "react";
import UserPicker from "./UserPicker"; export default function Header() {
return (
<div className="container">
<div className="row">
<div className="col-sm">
<button type="button" className="btn btn btn-light">Booking</button>
<button type="button" className="btn btn btn-light">BookList</button>
</div>
<div className="col-sm">
<UserPicker></UserPicker>
</div>
</div>
</div>
)
}

  UserPicker.js

const users = ['', 'sam', 'jason'];

export default function UserPicker() {

    // onChange={handleSelect}
return (
<select value={'sam'}>
{users.map(name => (
<option key={name} value={name}>{name}</option>
))}
</select>
);
}v

  Detials.js

const booking = [
{
"name": 'sam',
"title": "Meeting Room",
},
{
"name": 'json',
"title": "Lecture Hall",
}
] export default function Details() { return (
<div className="booking-details" style={{marginLeft: '120px'}}>
<h5 className="card-title"> Booking Details </h5>
<p className="card-text">{booking[0].name} {booking[0].title}</p>
</div>
);
}

  UserPicker和Details共享user数据,且在UserPicker中,可以选择user。因此在最外层的组件App中,要设一个user状态和改变user的setUser,让这两个数据共享。使用React.createContext()创建一个context,它接受一个可选的参数,就是共享数据的默认值,当然可以不传,直接调用createContext() 方法。不过,建议写上参数,把它看作是共享数据的格式定义,一看到这个context, 就知道要共享什么。那创建的context 放在什么地方呢?context 可以定义在任意位置,在src 目录下新建一个文件UserContext.js 来存放context.

import React from 'react';

export const UserContext = React.createContext({
user: '',
setUser: () => {}
})

  使用provider(<UserContext.Provider />组件)来提供共享数据,在App.js中,使用userState,然后把user和setUser放到Provider的value属性中,最后使用Provider把UserPicker和Details组件包起来

import { UserContext } from './UserContext';

export default function App() {
const [user, setUser] = useState(''); return (
<UserContext.Provider value={{user, setUser}}>
<div className="card">
<div className="card-header">
<Header></Header>
</div>
<div className="card-body">
<Details></Details>
</div>
</div>
</UserContext.Provider>
)
}

  UserPicker使用共享数据,最开始的时候,是使用Consumer组件。Consumer 组件的内容是一个函数表达式,函数的参数,就是Consumer组件帮我们注入到组件中的共享数据,它这时就可以直接使用共享数据了。

 
 import { UserContext } from './UserContext';
export default function UserPicker() {
return (
<UserContext.Consumer>
{(context) => {
const {user, setUser} = context;
return (
<select value={user} onChange={(e) => setUser(e.target.value)}>
{users.map(name => (
<option key={name} value={name}>{name}</option>
))}
</select>
)
}}
</UserContext.Consumer>
);
}

  Details中使用共享数据

import { UserContext } from './UserContext';

export default function Details() {

  return (
<UserContext.Consumer>
{ (context) => {
const { user } = context;
const seleted = booking.filter(data => data.name === user); return (
<div className="booking-details" style={{ marginLeft: '120px' }}>
<h5 className="card-title"> Booking Details </h5>
<p className="card-text">
{seleted.length > 0 &&
<React.Fragment>
{seleted[0].name} {seleted[0].title}
</React.Fragment>
}
</p>
</div>
)
}}
</UserContext.Consumer>
);
}

  如果使用的是React 16.8以上,可以使用useContext hooks,UserPicker.js

import { useContext } from 'react';
import { UserContext } from './UserContext'; export default function UserPicker() {
const {user, setUser} = useContext(UserContext); return (
<select value={user} onChange={(e) => setUser(e.target.value)}>
{users.map(name => (
<option key={name} value={name}>{name}</option>
))}
</select>
);
}

  Details.js

import { useContext } from 'react';
import { UserContext } from './UserContext'; export default function Details() { const { user } = useContext(UserContext);
const seleted = booking.filter(data => data.name === user); return (
<div className="booking-details" style={{ marginLeft: '120px' }}>
<h5 className="card-title"> Booking Details </h5>
<p className="card-text">
{seleted.length > 0 &&
<React.Fragment>
{seleted[0].name} {seleted[0].title}
</React.Fragment>
}
</p>
</div>
)
}

  这也会带来一个问题,当user 改变的时候,App所有子组件都会重新渲染,能不能只渲染consumer组件?要使用children属性,把要获取数据的子组件作为children传递给Provider. 定制一个Provider组件,接受children作为参数。UserContext.js

import React from 'react';
import {createContext, useState} from "react"; export const UserContext = createContext({
user: '',
setUser: () => {}
}) export const UserProvider = ({children}) => {
const [user, setUser] = useState(''); return <UserContext.Provider value={{user, setUser}}>
{children}
</UserContext.Provider>
}

  App.js

import { UserProvider } from './UserContext';

export default function App() {

    return (
<UserProvider>
<div className="card">
<div className="card-header">
<Header></Header>
</div>
<div className="card-body">
<Details></Details>
</div>
</div>
</UserProvider>
)
}

  当在userPicker中调用setUser改变user时,UserProvider组件会重新渲染,但是UserProvider的children不会重新渲染,因为children并没有改变,只有消费context的组件(children)才会重新渲染。UserProvider是从props中获取children, 更新组件内部的state并不会改变props。当后代组件中调用setUser, children的identity并不会改变。它和状态改变以前是同一个object,所以没有必要重新渲染children. 但context的consumer 并不一样,当Provider的值改变时,consumer 会重新渲染。Any components that consume the context, however, do re-render in response to the change of value on the provider, not because the whole tree of components has re-rendered.

  但使用object作为context的值,好不好?是不是把所有的共享数据都放到一个对象中?

value = {
theme: "lava",
user: 'sam',
language: "en"
};
<Context.Provider value={value}><App/></Context.Provider>

  在子组件中,有的组件消费theme, 有的组件消费user, 有的组件消费language。如果value中有一个值改变,所有consuer都会重新渲染,没有必要。可以使用多个provider

<ThemeContext.Provider value="lava">
<UserContext.Provider value="sam">
<LanguageContext.Provider value="en">
<App />
</LanguageContext.Provider>
</UserContext.Provider>
</ThemeContext.Provider>

  子组件,可以只想获取自己想要的值。

function InfoPage(props) {
const theme = useContext(ThemeContext);
const language = useContext(LanguageContext);
return (/* UI */);
} function Messages(props) {
const theme = useContext(ThemeContext);
const user = useContext(UserContext);
// subscribe to messages for user
return (/* UI */);
}

  定制一个AppProvider

function AppProvider({ children }) {
// maybe manage some state here
return (
<ThemeContext.Provider value="lava">
<UserContext.Provider value="sam">
<LanguageContext.Provider value="en">
{children}
</LanguageContext.Provider>
</UserContext.Provider>
</ThemeContext.Provider >
);
}

  使用它

<AppProvider>
<App/>
</AppProvider>

  如果再仔细一点,user和setUser可以作两个provider

import { createContext, useState } from "react";

export const UserContext = createContext({
user: ''
})
export const UserSetContext = createContext({
setUser: () => { }
}); export function UserProvider({ children }) {
const [user, setUser] = useState(null); return (
<UserContext.Provider value={user}>
<UserSetContext.Provider value={setUser}>
{children}
</UserSetContext.Provider>
</UserContext.Provider>);
}

  有时,也不一定非要使用Context,React官网还提供了组件组合

export default function App({ user }) {
const { username, avatarSrc } = user; return (
<main>
<Navbar username={username} avatarSrc={avatarSrc} />
</main>
);
} function Navbar({ username, avatarSrc }) {
return (
<nav>
<Avatar username={username} avatarSrc={avatarSrc} />
</nav>
);
} function Avatar({ username, avatarSrc }) {
return <img src={avatarSrc} alt={username} />;
}

  只有在App组件中知道Avatar组件的属性,可以直接创建一个Avatar组件向下传递

export default function App({ user }) {
const { username, avatarSrc } = user; const avatar = <img src={avatarSrc} alt={username} />; return (
<main>
<Navbar avatar={avatar} />
</main>
);
} function Navbar({ avatar }) {
return <nav>{avatar}</nav>;
}

React Context 的使用的更多相关文章

  1. React context基本用法

    React的context就是一个全局变量,可以从根组件跨级别在React的组件中传递.React context的API有两个版本,React16.x之前的是老版本的context,之后的是新版本的 ...

  2. [React] Prevent Unnecessary Rerenders of Compound Components using React Context

    Due to the way that React Context Providers work, our current implementation re-renders all our comp ...

  3. React Context API

    使用React 开发程序的时候,组件中的数据共享是通过数据提升,变成父组件中的属性,然后再把属性向下传递给子组件来实现的.但当程序越来越复杂,需要共享的数据也越来越多,最后可能就把共享数据直接提升到最 ...

  4. 探索 Redux4.0 版本迭代 论基础谈展望(对比 React context)

    Redux 在几天前(2018.04.18)发布了新版本,6 commits 被合入 master.从诞生起,到如今 4.0 版本,Redux 保持了使用层面的平滑过渡.同时前不久, React 也从 ...

  5. React Hooks +React Context vs Redux

    React Hooks +React Context vs Redux https://blog.logrocket.com/use-hooks-and-context-not-react-and-r ...

  6. [译]React Context

    欢迎各位指导与讨论 : ) 前言 由于笔者英语和技术水平有限,有不足的地方恳请各位指出.我会及时修正的 O(∩_∩)O 当前React版本 15.0.1 时间 2016/4/25 正文 React一个 ...

  7. react context跨组件传递信息

    从腾讯课堂看到的一则跨组件传递数据的方法,贴代码: 使用步骤: 1.在产生参数的最顶级组建中,使用childContextTypes静态属性来定义需要放入全局参数的类型 2.在父组件中,提供状态,管理 ...

  8. [译]迁移到新的 React Context Api

    随着 React 16.3.0 的发布,context api 也有了很大的更新.我已经从旧版的 api 更新到了新版.这里就分享一下我(作者)的心得体会. 回顾 下面是一个展示如何使用旧版 api ...

  9. React Context(一):隐式传递数据

    一 Context概述 Context provides a way to pass data through the component tree without having to pass pr ...

  10. react context toggleButton demo

    //toggleButton demo: //code: //1.Appb.js: import React from 'react'; import {ThemeContext, themes} f ...

随机推荐

  1. Linux 环境下安装redis

    目录 方法一 编辑安装 1.官网找到下载地址: 2.到目录下载redis安装包 3.下载后解压 4.安装工具 tcl (可以跳过) 5.编译安装 6.修改配置文件 7. 客户端连接 8.配置环境变量 ...

  2. 化繁为简|AIRIOT智慧水务信息化建设解决方案

    ​ "生产自动化,管理信息化"是现代化水厂建设的目标之一,需要在水质要求.工艺.生产.管理.环境等监测方面达到精细化管理标准,这是一个高度智能化,实现化繁为简智慧进阶的工程.传统水 ...

  3. 利用英特尔 Gaudi 2 和至强 CPU 构建经济高效的企业级 RAG 应用

    检索增强生成 (Retrieval Augmented Generation,RAG) 可将存储在外部数据库中的新鲜领域知识纳入大语言模型以增强其文本生成能力.其提供了一种将公司数据与训练期间语言模型 ...

  4. Linux运维面试总结

    1.Linux系统相关日志 /var/log/message:系统信息日志,包含错误信息 /var/log/secure:系统登录日志 /var/log/maillog:邮件日志 /var/log/c ...

  5. 利用docker 搭建File Browser 文件管理系统

    File Browser就是一个文件浏览器,因为linux并不方便桌面管理,所以Filebrowser就是帮助我们管理linux服务器上文件的程序,你可以称他为网盘程序,可以管理文件.可以分享文件,另 ...

  6. Python实现五子棋人机对战的二次开发

    Python实现人机对战的二次开发     在网上找到了一个使用python实现五子棋游戏,其中通过加入一个简单的AI算法实现了人机对战的功能,我觉得这个人机对战还是蛮有意思的,下面我分析一下五子棋游 ...

  7. redis 基础管理

    配置文件 优化redis配置文件定制 cat /nosql/redis/6379/redis.conf daemonize yes port 6379 logfile /nosql/redis/637 ...

  8. 异步长传文件插件 jquery validate 前端验证

    异步文件上传插件:http://fex.baidu.com/webuploader/getting-started.html html5 FormData 可以实现无刷新上传  有空了解 FileRe ...

  9. 鸿蒙HarmonyOS实战-ArkTS语言基础类库(通知)

    前言 移动应用中的通知是指应用程序发送给用户的一种提示或提醒消息.这些通知可以在用户设备的通知中心或状态栏中显示,以提醒用户有关应用程序的活动.事件或重要信息. 移动应用中的通知可以分为两种类型:本地 ...

  10. AGC055

    AGC055 第一次打AGC,好难受. T1 看了一眼题解,没看懂--但是还是做出来了. T2 感觉比 T1 简单,构造很好猜. 其他的没时间思考,T1 花了我 2h30min,难受. A.ABC I ...