react中的useContext--为什么使用(一)

React 的数据传递流程

在 React 中,数据传递通常是自上而下的,也就是父组件把数据通过 props 传递给子组件,子组件无法直接修改父组件的数据。
在这里插入图片描述

例子:父组件向子组件传递数据
const Parent = () => {const user = { name: "John", age: 24 };return <Child user={user} />;
};const Child = ({ user }) => {return (<div><h2>姓名: {user.name}</h2><h2>年龄: {user.age}</h2></div>);
};

问题不大,看起来很简单,对吧?


子组件传递数据的“噩梦”

但如果子组件需要修改父组件的数据,就必须通过回调函数,让子组件把新的数据“反向”传回父组件,让父组件更新数据。

例子:子组件修改父组件数据
const Parent = () => {const [user, setUser] = useState({ name: "John", age: 24 });// 让子组件调用这个方法来修改 userconst updateUser = (newName) => {setUser({ ...user, name: newName });};return <Child user={user} updateUser={updateUser} />;
};const Child = ({ user, updateUser }) => {return (<div><h2>姓名: {user.name}</h2><button onClick={() => updateUser("Alice")}>改名为 Alice</button></div>);
};

这样,点击按钮后,John 就会变成 Alice,但这还不算太糟糕。

真正的问题是:如果有很多层嵌套怎么办?


多层嵌套时的“数据传递地狱”

如果 Child 不是直接在 Parent 里面,而是嵌套了好几层,每一层都要手动传 props,就会变得非常痛苦。

数据传递“地狱”示例
const Parent = () => {const [user, setUser] = useState({ name: "John", age: 24 });const updateUser = (newName) => {setUser({ ...user, name: newName });};return <Level1 user={user} updateUser={updateUser} />;
};// 一层又一层地传递 props...
const Level1 = ({ user, updateUser }) => <Level2 user={user} updateUser={updateUser} />;
const Level2 = ({ user, updateUser }) => <Level3 user={user} updateUser={updateUser} />;
const Level3 = ({ user, updateUser }) => <Level4 user={user} updateUser={updateUser} />;
const Level4 = ({ user, updateUser }) => (<div><h2>姓名: {user.name}</h2><button onClick={() => updateUser("Alice")}>改名为 Alice</button></div>
);

问题:

  • 你要在 每一层组件 都写 props 传递,代码变得冗长且难以维护。
  • 组件越多,数据传递越混乱,很容易出错。
  • 这个问题被称为 “Props Drilling”(属性挖掘),就像挖矿一样,数据要一层一层往下挖。

如何解决?—— useContext 来救场!

React 提供了 useContext,它就像一个全局数据仓库,可以让任何组件直接访问数据,而不需要层层 props 传递。

步骤 1:创建 Context
import { createContext, useState } from "react";// 1. 创建 Context
const UserContext = createContext();// 2. 创建 Provider 组件
const UserProvider = ({ children }) => {const [user, setUser] = useState({ name: "John", age: 24 });const updateUser = (newName) => {setUser({ ...user, name: newName });};return (<UserContext.Provider value={{ user, updateUser }}>{children}  {/* 这里的 children 让所有子组件都能访问这个 Context */}</UserContext.Provider>);
};export { UserContext, UserProvider };

步骤 2:子组件直接用 useContext 读取数据
import { useContext } from "react";
import { UserContext } from "./UserContext";const UserProfile = () => {const { user, updateUser } = useContext(UserContext);return (<div><h2>姓名: {user.name}</h2><button onClick={() => updateUser("Alice")}>改名为 Alice</button></div>);
};export default UserProfile;

步骤 3:在 App.js 里包裹 Provider
import React from "react";
import { UserProvider } from "./UserContext";
import UserProfile from "./UserProfile";const App = () => {return (<UserProvider><UserProfile /></UserProvider>);
};export default App;

为什么 useContext 很强大?

  1. 避免了“数据传递地狱”:不需要层层 props 传递,所有组件都能直接访问数据。
  2. 代码更清晰:不管组件嵌套多少层,都能方便地读取和更新数据。
  3. 性能更好:不会因为 props 变化导致所有中间组件都重新渲染。

总结

  1. 传统数据传递方式(props

    • 适合小型项目,数据传递简单时使用。
    • 但是当层级变深时,props drilling 让代码变得难以维护。
  2. useContext 方式

    • 适合共享状态的场景,比如用户信息、主题设置、语言切换等。
    • 让所有组件都能直接访问数据,避免 props 层层传递。
  3. 最佳实践

    • 如果数据只在父子组件之间传递,用 props 即可。
    • 如果数据需要被多个组件共享,使用 useContext 来简化代码。

🚀 现在,你可以摆脱“数据传递地狱”,用 useContext 让 React 代码更清爽!

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

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

相关文章

如何使用 LLM 生成的术语自动在搜索应用程序上构建 autocomplete 功能

作者&#xff1a;来自 Elastic Michael Supangkat 了解如何在 Elastic Cloud 中&#xff0c;通过使用 LLM 生成的词汇&#xff0c;为搜索应用增强自动补全功能&#xff0c;实现更智能、更动态的搜索建议。 自动补全是搜索应用中的一项关键功能&#xff0c;它通过在用户输入时实…

MAVEN手动配置(阿里云)全教程

介于网上各种各样的MAVEN配置过程中方法大致相同却细节参差不齐&#xff0c;我总结了我遇见的一些问题&#xff0c;来完全的解决MAVEN手动配置的全过程&#xff0c;以及分享解决小毛病的经验。 所需材料&#xff1a; MAVEN3.9.9&#xff08;下载适合自己的版本即可&#xff09…

DeepSeek 3FS:端到端无缓存的存储新范式

在 2025 年 2 月 28 日&#xff0c;DeepSeek 正式开源了其高性能分布式文件系统 3FS【1】&#xff0c;作为其开源周的压轴项目&#xff0c;3FS 一经发布便引发了技术圈的热烈讨论。它不仅继承了分布式存储的经典设计&#xff0c;还通过极简却高效的架构&#xff0c;展现了存储技…

HarmonyOS:如何将图片转为PixelMap并进行图片缓存策略

前言&#xff1a;在HarmonyOS项目开发中&#xff0c;我们使用Ark-Ts语言开发项目。我们有个功能是拍照&#xff0c;除了正常显示出来&#xff0c;并且上传服务器。我在开发过程中&#xff0c;遇到的问题是&#xff0c;如果离开这个页面再回到当前页面仍要显示图片&#xff0c;那…

2025.3.9机器学习笔记:文献阅读

2025.3.9周报 一、文献阅读题目信息摘要Abstract创新点网络架构实验结论不足以及展望 一、文献阅读 题目信息 题目&#xff1a; Time-series generative adversarial networks for flood forecasting期刊&#xff1a; Journal of Hydrology作者&#xff1a; Peiyao Weng, Yu …

linux固定IP并解决虚拟机无法ping其他电脑问题

linux固定IP并解决虚拟机无法ping其他电脑问题 1.找到网卡文件 vim /etc/sysconfig/network-scripts/ifcfg-ens33 2.编辑文件信息 BOOTPROTO 这个dhcp改为static#添加以下内容IPADDR<你的IP地址>NETMASK<子网掩码>&#xff0c;例如255.255.255.0。GATEWAY<网…

Spring实战spring-ai运行

目录 1. 配置 2 .搭建项目 3. 查看对应依赖 3.1 OpenAI 依赖 3.2 配置 OpenAI API 密钥 application.properties application.yml 4. openai实战 5. 运行和测试 6. 高级配置 示例&#xff1a;配置模型和参数 解释&#xff1a; 7. 处理异常和错误 示例&#xff1a;…

docker:配置 Docker 镜像加速器

1 镜像加速器介绍 默认情况下&#xff0c;将来从docker hub&#xff08;https://hub.docker.com/&#xff09;上下载docker镜像&#xff0c;太慢。一般都会配置镜像加速器&#xff1a; USTC&#xff1a;中科大镜像加速器&#xff08;https://docker.mirrors.ustc.edu.cn&…

[内网安全] Windows 本地认证 — NTLM 哈希和 LM 哈希

关注这个专栏的其他相关笔记&#xff1a;[内网安全] 内网渗透 - 学习手册-CSDN博客 0x01&#xff1a;SAM 文件 & Windows 本地认证流程 0x0101&#xff1a;SAM 文件简介 Windows 本地账户的登录密码是存储在系统本地的 SAM 文件中的&#xff0c;在登录 Windows 的时候&am…

算法-图-dijkstra 最短路径

理论知识 dijkstra三部曲 朴素版dijkstra 模拟过程 堆优化版dijksra 经典模版例题 Dijkstra求最短路 I 参加科学大会&#xff08;第六期模拟笔试&#xff09;--模版题 网络延迟 ref 理论知识 最短路是图论中的经典问题即&#xff1a;给出一个有向图&#xff0c;一…

Qt添加MySql数据库驱动

文章目录 一. 安装MySql二.编译mysql动态链接库 Qt版本&#xff1a;5.14.2 MySql版本&#xff1a;8.0.41 一. 安装MySql 参考这里进行安装&#xff1a;https://blog.csdn.net/qq_30150579/article/details/146042922 将mysql安装目录里的bin&#xff0c;include和lib拷贝出来…

浅论数据库聚合:合理使用LambdaQueryWrapper和XML

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、数据库聚合替代内存计算&#xff08;关键优化&#xff09;二、批量处理优化四、区域特殊处理解耦五、防御性编程增强 前言 技术认知点&#xff1a;使用 XM…

Ubuntu 22.04安装NVIDIA A30显卡驱动

一、安装前准备 1.禁用Nouveau驱动 Ubuntu默认使用开源Nouveau驱动&#xff0c;需要手动禁用&#xff1a; vim /etc/modprobe.d/blacklist-nouveau.conf # 添加以下内容&#xff1a; blacklist nouveau options nouveau modeset0 # 更新内核并重启&#xff1a; update-initr…

Docker Desktop 4.38 安装与配置全流程指南(Windows平台)

一、软件定位与特性 Docker Desktop 是容器化应用开发与部署的一体化工具&#xff0c;支持在本地环境创建、管理和运行Docker容器。4.38版本新增GPU加速支持、WSL 2性能优化和Kubernetes 1.28集群管理功能&#xff0c;适用于微服务开发、CI/CD流水线搭建等场景。 二、安装环境…

音视频入门基础:RTP专题(15)——FFmpeg源码中,获取RTP的视频信息的实现

一、引言 通过FFmpeg命令可以获取到SDP文件描述的RTP流的视频压缩编码格式、色彩格式&#xff08;像素格式&#xff09;、分辨率、帧率信息&#xff1a; ffmpeg -protocol_whitelist "file,rtp,udp" -i XXX.sdp 本文以H.264为例讲述FFmpeg到底是从哪个地方获取到这…

深度学习---卷积神经网络

一、卷积尺寸计算公式 二、池化 池化分为最大池化和平均池化 最常用的就是最大池化&#xff0c;可以认为最大池化不需要引入计算&#xff0c;而平均池化需要引出计算&#xff08;计算平均数&#xff09; 每种池化还分为Pooling和AdaptiveAvgPool Pooling(2)就是每2*2个格子…

netty中Future和ChannelHandler

netty中的Future&#xff0c;继承自 jdk中的Future&#xff0c;&#xff0c; jdk中的Future&#xff0c;很垃圾&#xff0c;只能同步阻塞获取结果&#xff0c;&#xff0c;&#xff0c; netty中的Future进行了升级&#xff0c;&#xff0c;可以addListener()异步获取结果&…

java 初学知识点总结

自己总结着玩 1.基本框架 public class HelloWorld{ public static void main(String[] args){ }//类名用大写字母开头 } 2.输入&#xff1a; (1)Scanner:可读取各种类型&#xff0c;字符串相当于cin>>; Scanner anew Scanner(System.in); Scan…

质量属性场景描述

为了精确描述软件系统的质量属性&#xff0c;通常采用质量属性场景&#xff08;Quality Attribute Scenario&#xff09;作为描述质量属性的手段。质量属性场景是一个具体的质量属性需求&#xff0c;使利益相关者与系统的交互的简短陈述。 质量属性场景是一种用于描述系统如何…

数据可携带权的多重价值与实践思考

文章目录 前言一、数据可携带权的提出与立法二、数据可携带权的多重价值1、推动数据要素市场化配置2、促进市场竞争与创新3、强化个人数据权益 三、数据可携带权的实践挑战1、数据安全与隐私保护面临风险2、接口差异导致数据迁移成本高昂3、可携带的数据范围尚存争议 数据可携带…