React16源码: context用法与createContext源码实现

context


1 )概述

  • 在react的设计中,组件与组件之间的通信通常是
    • 父组件通过 props 给子组件传递子组件需要的属性
    • 父组件通过传递一些回调函数给子组件
    • 让子组件在某些特定的时候,可以调用一些父组件的特性
  • 这种情况,会存在一个问题
    • 就是react的应用中组件和组件之间并不一定只有父子关系
    • 还会存在着像父子嵌套多层之后,第一层和最下层的组件
    • 他们之间是一个主孙的一个关系
    • 他们中间会隔着好几层不同的组件,如果通过props进行一个传递,是不太现实的
    • 还有就是中间的那几层组件,不一定是我们自己写的
    • 中间的组件要去传递这个 props,其实是完全没有意义的事情
  • 所以,react 提供了一个 context 的一个使用方式
    • 在上级组件中,我们提供了一个 context 对象之后
    • 只要是在它下面渲染的组件都可以通过 context 这个属性去获取到它提供的这部分内容
    • 以此达到一个跨越多层组件传递信息的一个功能
  • context 有两种实现方式
    • 第一种,是通过老的 context 的 API 叫做 childContextTypes
      • 老的 childContextTypes 在react17 这个大版本发布的时候被废弃
      • 历史原因,用的还挺多
    • 第二种, 是通过新版提供的 createContext 这个API
      • 这块下面会分析下源码

用法示例

1 )示例演示

这个示例,演示了 新旧 两个 api 的用法

import React from 'react'
import PropTypes from 'prop-types'const { Provider, Consumer } = React.createContext('default')// 定义一个父组件 作为 最外层
class Parent extends React.Component {state = {childContext: '123',newContext: '456',}// react 静态方法 apigetChildContext() {return { value: this.state.childContext, a: 'aaaaa' }}render() {return (<><div><label>childContext:</label><inputtype="text"value={this.state.childContext}onChange={e => this.setState({ childContext: e.target.value })}/></div><div><label>newContext:</label><inputtype="text"value={this.state.newContext}onChange={e => this.setState({ newContext: e.target.value })}/></div>{/* 基于 Provider来传递 */}<Provider value={this.state.newContext}>{this.props.children}</Provider></>)}
}// 定义第二个父组件 作为 中间层
class Parent2 extends React.Component {// { value: this.state.childContext, a: 'bbbbb' }getChildContext() {return { a: 'bbbbb' }}render() {return this.props.children}
}// 定义第一个子组件 内部使用 Consumer
function Child1(props, context) {console.log(context)return <Consumer>{value => <p>newContext: {value}</p>}</Consumer>
}// 声明子组件需要的 props
Child1.contextTypes = {value: PropTypes.string,
}// 定义第二个子组件
class Child2 extends React.Component {render() {return (<p>childContext: {this.context.value} {this.context.a}</p>)}
}// Child2.contextType = Consumer
// 声明 子组件2 需要的 props
Child2.contextTypes = {value: PropTypes.string,a: PropTypes.string,
}// 父组件不声明,子组件无法获取 props
Parent.childContextTypes = {value: PropTypes.string,a: PropTypes.string,
}// 父组件不声明,子组件无法获取 props
Parent2.childContextTypes = {a: PropTypes.string,
}// 最终的组件树,不同组件的嵌套
export default () => (<Parent><Parent2><Child1 /><Child2 /></Parent2></Parent>
)

2 )关于 childContextTypes 旧版API的说明

  • 主要以 Child2 组件来说明

  • 上级组件中声明这个 getChildContext 这个方法

  • 然后return的这个对象, 就是作为子组件当中能够获取这个 context 的对象

  • 但是有一点必须要注意,就是父组件必须要声明 childContextTypes, 即: Parent.childContextTypes

      Parent.childContextTypes = {value: PropTypes.string,a: PropTypes.string,}
    
  • 上层组件是必须要声明的

  • 如果不声明,它提供的这个 context ,子组件是无法获取到的

  • 想要获取上层组件提供的 context,需要在子组件 Child2 当中

  • 声明自己需要的 contextTypes, 例如

     Child2.contextTypes = {value: PropTypes.string,a: PropTypes.string,}
    
  • 它的内容也是跟上层组件的 childContextTypes 是一样

  • 两者区别是: 有或没有子组件

  • 在这个渲染的过程中,比如这个 Child2 组件,希望获取父层组件中提供的 context 里面的某几个属性

  • Chid 自己就需要去声明使用几个属性,为何这么做呢

    • 在react当中,它的上层组件不一定只有一个
    • 它上层组件中提供的 context 也不一定只有一个
    • 它们是会有一个 merge 的一个过程的
    • 所以在很多属性中 react 要知道你想要获取上层组件当中提供的context里面的哪几个属性
    • 所以要通过这种方式进行一个声明

3 )关于 createContext 新版API的分析

  • 通过 react.createContext,它返回了一个对象
    const { Provider, Consumer } = React.createContext('default')
    
  • 这个对象里面包含了 ProviderConsumer,是一个 context 提供方和 context 的订阅方
  • 这两个都是组件,我们通过在 Parent 这边我们提供了Provider, 然后上面指定 value
    <Provider value={this.state.newContext}>{this.props.children}</Provider>
    
    • 这个 value 就是 context,即提供的context的属性信息
    • 在它的子树下面,只需要在想要用到context的地方
    • 通过这个 Consumer 组件 (它传入的是一个回调方法)
    • 这个方法是一个function Component,它接收它的一个value
    • 并且把这个想要渲染的东西渲染出来就可以了
      // 声明子组件需要的 props
      Child1.contextTypes = {value: PropTypes.string,
      }
      
  • 所以ProviderConsumer 是一一对应的关系
  • 在上层组件里面定义之后,子组件里面你想要在哪个地方用到这个属性
  • 你再去专门用这个组件去进行一个渲染就可以了

4 )为什么要弃用老的API要改成这种新的API

  • 因老的API它对于context提供方,它下层的所有组件的影响太大了
  • 它会导致它下层的所有组件, 即便在没有任何更新的情况下,它每一次更新的过程当中
  • 仍然要进行完整的渲染,所以对性能的损耗会非常大

createContext 源码分析

定位到 reactContext.js

/*** Copyright (c) Facebook, Inc. and its affiliates.** This source code is licensed under the MIT license found in the* LICENSE file in the root directory of this source tree.** @flow*/import {REACT_PROVIDER_TYPE, REACT_CONTEXT_TYPE} from 'shared/ReactSymbols';import type {ReactContext} from 'shared/ReactTypes';import warningWithoutStack from 'shared/warningWithoutStack';
import warning from 'shared/warning';export function createContext<T>(defaultValue: T,calculateChangedBits: ?(a: T, b: T) => number,
): ReactContext<T> {if (calculateChangedBits === undefined) {calculateChangedBits = null;} else {if (__DEV__) {warningWithoutStack(calculateChangedBits === null ||typeof calculateChangedBits === 'function','createContext: Expected the optional second argument to be a ' +'function. Instead received: %s',calculateChangedBits,);}}const context: ReactContext<T> = {$$typeof: REACT_CONTEXT_TYPE,_calculateChangedBits: calculateChangedBits,// As a workaround to support multiple concurrent renderers, we categorize// some renderers as primary and others as secondary. We only expect// there to be two concurrent renderers at most: React Native (primary) and// Fabric (secondary); React DOM (primary) and React ART (secondary).// Secondary renderers store their context values on separate fields._currentValue: defaultValue,_currentValue2: defaultValue,// These are circularProvider: (null: any),Consumer: (null: any),};context.Provider = {$$typeof: REACT_PROVIDER_TYPE,_context: context,};let hasWarnedAboutUsingNestedContextConsumers = false;let hasWarnedAboutUsingConsumerProvider = false;if (__DEV__) {// A separate object, but proxies back to the original context object for// backwards compatibility. It has a different $$typeof, so we can properly// warn for the incorrect usage of Context as a Consumer.const Consumer = {$$typeof: REACT_CONTEXT_TYPE,_context: context,_calculateChangedBits: context._calculateChangedBits,};// $FlowFixMe: Flow complains about not setting a value, which is intentional hereObject.defineProperties(Consumer, {Provider: {get() {if (!hasWarnedAboutUsingConsumerProvider) {hasWarnedAboutUsingConsumerProvider = true;warning(false,'Rendering <Context.Consumer.Provider> is not supported and will be removed in ' +'a future major release. Did you mean to render <Context.Provider> instead?',);}return context.Provider;},set(_Provider) {context.Provider = _Provider;},},_currentValue: {get() {return context._currentValue;},set(_currentValue) {context._currentValue = _currentValue;},},_currentValue2: {get() {return context._currentValue2;},set(_currentValue2) {context._currentValue2 = _currentValue2;},},Consumer: {get() {if (!hasWarnedAboutUsingNestedContextConsumers) {hasWarnedAboutUsingNestedContextConsumers = true;warning(false,'Rendering <Context.Consumer.Consumer> is not supported and will be removed in ' +'a future major release. Did you mean to render <Context.Consumer> instead?',);}return context.Consumer;},},});// $FlowFixMe: Flow complains about missing properties because it doesn't understand definePropertycontext.Consumer = Consumer;} else {context.Consumer = context;}if (__DEV__) {context._currentRenderer = null;context._currentRenderer2 = null;}return context;
}
  • 定位到 createContext, 它接收的是一个 defaultValue 和 calculateChangeBits方法
    • calculateChangeBits方法是用来计算新老context它们的一个变化的
  • 在这里,它声明了一个context对象, 这个对象跟之前的 ReactElement 非常的像
    • 它有一个 $$typeof, 这里的 $$typeof 跟 ReactElement 里面的 $$typeof 是不一样的
  • 下面有这两个属性 _currentValue, _currentValue2
    • 这两个属性它们的用处是一样的,使用的地方会不一样,比如说,不同平台里面会不一样
    • 它的 _currentValue 是用来记录 Provider上面提供的这个 value
    • 在有变化的情况下,它就会更新到这个 _currentValue 上面,这就是用来记录最新的context的值的
  • 下面会有一个 Provider和 Consumer
  • 然后再接下去, 有一个 context.Provider 这个对象
     context.Provider = {$$typeof: REACT_PROVIDER_TYPE,_context: context,};
    
    • 这个_context指向这个context的对象
  • 忽略DEV相关的判断代码
  • 在最后我们看到 context.Consumer = context
    • 也就是说 Consumer是指向这个对象自己的
    • 在Consumer进行渲染的时候,它要去获取这个value
    • 直接从它本身上面去获取到这个 _currentValue
    • 就可以拿到最新的 context 的值了,然后再调用Consumer 的回调方法把它传进去
    • 就可以渲染出最新的内容
  • 所以这就是它的一个基本的实现原理。
  • 整个context 新的contextAPI的一个源码不是特别复杂,但是整个通信过程并不止这一点
  • 在这里,主要是弄清楚 Provider 和 Consumer 的关系
  • 需要注意的是
    • 返回的这个 Provider 和 Consumer,它们里面都有 $$typeof
    • 它们并不是用替代 ReactElement 里面的 $$typeof
    • 因为它返回的这个对象是整体, 是作为 ReactElement 里面的 type 属性去存储的
    • 所以跟这边的 $$typeof 是完全没有任何关系的
    • 也就是说 type 里面它还有一个 $$typeof, 表明它是一个context的 Provider,或 Consumer
    • 这个问题在前文也有提过

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/591311.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring ProxyFactoryBean

Spring的ProxyFactoryBean是一个非常有用的工具&#xff0c;它可以帮助我们创建代理对象&#xff0c;以实现AOP&#xff08;面向切面编程&#xff09;等功能。本文将介绍ProxyFactoryBean的基本概念、使用方法和注意事项。 ProxyFactoryBean的设计模式 ProxyFactoryBean是Spri…

谷歌AI模型Gemini被指演示视频造假 /深圳园区推出鸿蒙工程师租房优惠 |魔法半周报

我有魔法✨为你劈开信息大海❗ 高效获取AIGC的热门事件&#x1f525;&#xff0c;更新AIGC的最新动态&#xff0c;生成相应的魔法简报&#xff0c;节省阅读时间&#x1f47b; &#x1f525;资讯预览 谷歌AI模型Gemini被指演示视频造假&#xff0c;AI“神话”成“笑话” 深圳园…

TCP/IP的网络层(即IP层)之IP地址和网络掩码,在视频监控系统中的配置和应用

在给客户讲解我们的AS-V1000视频监控平台的时候&#xff0c;有的客户经常会配置错误IP地址的掩码和网关&#xff0c;导致出现一些网路问题。而在视频监控系统中&#xff0c;IP地址和子网掩码是用于标识网络中设备的重要标识符。IP地址被用来唯一地标识一个网络设备&#xff0c;…

Docker:部署若依前后端分离版

Docker&#xff1a;部署若依前后端分离版 1. 停止天翼云上的原来跑的若依项目2. 停止腾讯云上的若依项目3. 使用Docker部署3.1 天翼云数据库&Redis3.1.1 部署数据库3.1.2 部署Redis数据库3.1.1 部署Nginx(这里被天翼云坑了换的腾讯云运行nginx) 3.2 腾讯云部署后端&前端…

高性价比LDR6028Type-C转3.5mm音频和PD快充转接器

随着市面上的大部分手机逐渐取消了3.5mm音频耳机接口&#xff0c;仅保留一个Type-C接口&#xff0c;追求音质和零延迟的用户面临着一大痛点。对于这些用户&#xff0c;Type-C转3.5mm接口线的出现无疑是一大福音。这款线材在刚推出时就受到了手机配件市场的热烈欢迎&#xff0c;…

linux 测速 speedtest

Speedtest CLI: Internet speed test for the command line ubuntu、debian 非root加sudo sudo apt-get install curl -y curl -s https://packagecloud.io/install/repositories/ookla/speedtest-cli/script.deb.sh | sudo bash sudo apt-get install speedtest -yroot用户不…

提供电商Api接口-100种接口,淘宝,1688,抖音商品详情数据安全,稳定,支持高并发

Java是一种高级编程语言&#xff0c;由Sun Microsystems公司于1995年推出&#xff0c;现在属于Oracle公司开发和维护。Java以平台无关性、面向对象、安全性、可移植性和高性能著称&#xff0c;广泛用于桌面应用程序、嵌入式系统、企业级服务、Android移动应用程序等。 接口是Ja…

编程语言的进化:智能化与多样化的未来

作为沟通人类与机器的桥梁&#xff0c;编程语言的发展一直是技术进步的重要推动力。在技术的金字塔上&#xff0c;编程语言作为软件开发的基石&#xff0c;其每一次革新都将引领着信息时代的下一个潮流。从早期的机器码&#xff0c;到现代的高级语言&#xff0c;编程语言的进化…

【LeetCode每日一题】1599. 经营摩天轮的最大利润(模拟)—2024新年快乐!

2024-1-1 文章目录 [1599. 经营摩天轮的最大利润](https://leetcode.cn/problems/maximum-profit-of-operating-a-centennial-wheel/)思路&#xff1a; 1599. 经营摩天轮的最大利润 思路&#xff1a; 1.对摩天轮的运转情况进行模拟&#xff0c; 2.遍历数组&#xff0c;分别计…

《2023我的编程之旅》

在2023年&#xff0c;我经历了一段充满挑战与成长的编程之旅。这一年&#xff0c;我不仅在技术上取得了显著的进步&#xff0c;也在职业规划与心灵成长方面有了更多的认识。现在&#xff0c;我想通过这篇文章&#xff0c;分享我的经历、感悟和未来的规划。 一、印象深刻的实战…

一个Oracle数据库可以有多个DBID吗?

一个新建的数据库&#xff0c;我在分析 dba_hist_sql_plan 表时&#xff0c;发现其中有2个DBID。 select distinct dbid from dba_hist_sql_plan;DBID ---------- 1899454952 1467201108而且两个DBID的记录都很多&#xff1a; SQL> select count(*) from dba_hist_sql_pla…

开放路径最短优先协议OSPF基础

开放路径最短优先协议OSPF基础 对比RIP 对比距离矢量路由协议(RIP)&#xff0c;OSPF协议交换的不是路由条目&#xff0c;而是链路信息&#xff0c;并通过SPF算法计算出最佳路由&#xff0c;链路状态信息内含有路由接口、IP地址、掩码、cost值等&#xff0c;进而形成了链路状态…

SQL之CASE WHEN用法详解

目录 一、简单CASE WHEN函数&#xff1a;二、CASE WHEN条件表达式函数三、常用场景 场景1&#xff1a;不同状态展示为不同的值场景2&#xff1a;统计不同状态下的值场景3&#xff1a;配合聚合函数做统计场景4&#xff1a;CASE WHEN中使用子查询场景5&#xff1a;经典行转列&am…

2023国货美妆品牌:年初“外攻”,年末“内守”

2023年&#xff0c;国货美妆品牌迎来全面爆发&#xff0c;多数品牌交了一份满分答卷。特别是双11期间&#xff0c;多家国货美妆品牌跻身前列&#xff0c;珀莱雅更是成功登顶榜首&#xff0c;打破了天猫美妆双11榜首多年来被国际大牌占领的局面。 双11天猫美容护肤TOP20天品牌榜…

一款新型霍尔板在推杆电机上的运用

目录 一、推杆总成的组成 二、霍尔板在推杆电机上的运用 推杆电机是一种旋转运动转变为电动推杆直线往复运动的电动驱动装置,可广泛运用于医疗、家具、家庭、电子、电力、机械等领域&#xff0c;主要由电机驱动&#xff0c;推杆总成、传动轴、控制箱组成。 一、推杆总成的组成…

搭建Python开发环境 Pycharm编程 + 嵌入 (保姆级教程)

搭建环境 这部分也比较简单&#xff0c;因为我们刚初始化的树莓派&#xff0c;就像一个婴儿一样&#xff0c;非常干净&#xff0c;所以流程很轻松。 建立远程连接后&#xff0c;终端输入以下指令&#xff1a; 更新树莓派 sudo apt-get update sudo apt-get upgrade -y 下载…

探索 PyTorch 中的 torch.nn 模块(2)

目录 torch.nn模块详解 register_module_forward_pre_hook 主要特性和用途 警告 钩子签名 使用方法 返回值 示例代码 register_module_forward_hook 主要特性和用途 警告 钩子签名 使用方法 参数 返回值 示例代码 register_module_backward_hook 主要用途 …

hive多分隔符外表支持

在hive 外表关联文本的时候 有时会遇到不是一个长度的分割符比如"~" 这种。这个时候使用shell命令多处理一步处理成单分隔符也可以&#xff0c;但是会有出错的风险。我们可以通过hive中指定的序列类来完成多分隔符的识别。 CREATE EXTERNAL TABLE text_mid1( id STRI…

使用Triton部署ONNX模型

介绍 适用于各种 AI 工作负载的推理&#xff1a;借助 NVIDIA Triton™&#xff0c;在任何处理器&#xff08;GPU、CPU 或其他&#xff09;上&#xff0c;对使用基于任何框架的&#xff0c;经过训练的机器学习模型或深度学习模型&#xff0c;进行推理部署。Triton 是 NVIDIA AI…

【C#】知识点实践序列之Lock简单解决并发引起数据重复问题

欢迎来到《小5讲堂之知识点实践序列》文章&#xff0c;大家好&#xff0c;我是全栈小5。 这是2023年第3篇文章&#xff0c;此篇文章是C#知识点实践序列文章&#xff0c;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 本篇在Lock锁定代码块基…