新闻中心
React组件间数据传递:从子组件向父组件通信的最佳实践

本文详细介绍了在react应用中,如何实现子组件向父组件传递数据。通过利用react的单向数据流特性,结合回调函数作为props和父组件的状态管理,可以安全有效地将子组件(如表单输入)的数据传递给父组件,进而触发数据请求等逻辑。
在React应用开发中,组件之间的数据流动是核心概念之一。常见的场景是父组件向子组件传递数据(通过props),但当子组件需要将其内部状态或事件产生的数据反馈给父组件时,就需要采用特定的模式。本文将以一个具体的例子,讲解如何将子组件InputField中的用户输入值传递给父组件App,并用于API请求。
理解React的数据流
React推崇单向数据流(Unidirectional Data Flow)。这意味着数据通常从父组件流向子组件。如果子组件需要影响父组件的状态,它不能直接修改父组件的变量。相反,子组件应该通过调用父组件传递给它的函数(作为props)来通知父组件进行状态更新。这种模式被称为“提升状态”(Lifting State Up)或“回调函数模式”。
实现子组件向父组件传递数据
我们将通过以下步骤实现数据传递:
- 修改子组件InputField,使其接受一个回调函数作为props。
- 在InputField的事件处理函数中,调用这个回调函数,并传入需要传递的数据。
- 修改父组件App,定义一个状态来存储从子组件接收到的数据。
- 在App中定义一个函数,作为回调函数传递给InputField。
- 在App的useEffect中,监听这个状态的变化,并触发相应的API请求。
1. 修改子组件:InputField.js
首先,我们需要让InputField组件能够接受一个名为onSubmit的prop,这个prop将是一个函数。当用户提交表单时,handleSubmit函数会获取输入值inputVal,然后调用这个onSubmit prop,并将inputVal作为参数传递出去。
// InputField.js
export default function InputField({ onSubmit }) { // 接受 onSubmit prop
function handleSubmit(e) {
e.preventDefault(); // 阻止浏览器重新加载页面
const form = e.target;
const inputVal = form.myInput.value; // 获取输入框的值
console.log("Input value from child:", inputVal);
// 调用父组件传入的回调函数,并将输入值传递给它
if (onSubmit) {
onSubmit(inputVal);
}
}
return (
<form method="post" onSubmit={handleSubmit}>
<input name="myInput" id="adress-field" placeholder="Enter address" autoComplete="on" />
<button type="submit" id="adress-button">Send</button>
</form>
);
}注意事项:
- InputField现在是一个受控组件,它通过props接收行为。
- onSubmit是一个约定俗成的prop名,表示当子组件内部发生“提交”事件时,父组件应该执行的逻辑。
2. 修改父组件:App.js
接下来,我们需要在App组件中:
- 使用useState定义一个状态变量,例如searchValue,用于存储从InputField接收到的地址值。
- 定义一个函数handleSearchSubmit,它将作为onSubmit prop传递给InputField。当InputField调用onSubmit时,handleSearchSubmit会被执行,并更新searchValue状态。
- 将InputField组件正确地渲染到DOM中,并传递handleSearchSubmit作为其onSubmit prop。
- 修改useEffect,使其依赖于searchValue状态,并在searchValue更新时触发API请求。
// App.js
import './App.css';
import AccountNumber from "./components/AccountNumber";
import InputField from "./components/InputField";
import { useEffect, useState } from "react";
function App() {
// 状态用于存储从InputField接收到的搜索值
const [searchValue, setSearchValue] = useState(''); // 初始化为空字符串,避免首次渲染时请求 'null'
// token fetch 相关的状态
const [tokens, setTokens] = useState([]);
const [loading, setLoading] = useState(false);
// 定义一个回调函数,用于接收InputField传递过来的值
const handleSearchSubmit = (val) => {
setSearchValue(val); // 更新searchValue状态
};
// useEffect 钩子用于处理副作用,例如数据获取
useEffect(() => {
// 只有当 searchValue 有效时才进行数据请求
if (searchValue) {
setLoading(true);
fetch(`https://api.multiversx.com/accounts/${searchValue}/tokens`)
.then(response => response.json())
.then(json => setTokens(json))
.catch(error => {
console.error("Error fetching tokens:", error);
setTokens([]); // 请求失败时清空或重置tokens
})
.finally(() => {
setLoading(false);
});
} else {
// 如果 searchValue 为空,可以清空tokens或显示提示
setTokens([]);
setLoading(false);
}
console.log("Current tokens state:", tokens); // 注意:这里可能打印的是旧的tokens状态
}, [searchValue]); // 依赖数组中包含 searchValue,当其改变时重新运行 effect
function round(nr, ten) {
return Math.round(nr * ten) / ten;
}
function numberWithSpaces(nr) {
return nr.toString().replace(/\B(?=(\d{3})+(?!\d))/g, " ");
}
return (
<content className="content">
<div className="up-side">
<div className="account-number-box">
<p id="p-account-number">Total number of accounts</p>
<p id="account-number"><AccountNumber/></p>
</div>
<div className="adress-search">
{/* 正确地渲染 InputField 组件,并传递回调函数 */}
<InputField onSubmit={handleSearchSubmit} />
</div>
{/* 可以显示当前的搜索值以供调试 */}
<p>Current Search Value: {searchValue}</p>
</div>
<div className="dow
n-side">
<table className="Token-section-output">
{loading ? (
<div>Loading...</div>
) : (
<>
<h1>Tokens</h1>
<table className='Token-section-output' border={0}>
<tr className='token-row-type'>
<th className='token-column'>Name</th>
<th className='center-column'>Price</th>
<th>Hold</th>
</tr>
<tr className="space20"/>
{tokens.map(token => (
<tr className='token-row' key={token.id}>
<td className='token-column'>
@@##@@
<p>{token.name}</p>
<div class="aritcle_card">
<a class="aritcle_card_img" href="/ai/1247">
<img src="https://img.php.cn/upload/ai_manual/000/000/000/175680148166020.png" alt="秀脸FacePlay">
</a>
<div class="aritcle_card_info">
<a href="/ai/1247">秀脸FacePlay</a>
<p>一款集成AI换脸、照片跳舞等多种AI特效玩法的App</p>
<div class="">
<img src="/static/images/card_xiazai.png" alt="秀脸FacePlay">
<span>124</span>
</div>
</div>
<a href="/ai/1247" class="aritcle_card_btn">
<span>查看详情</span>
<img src="/static/images/cardxiayige-3.png" alt="秀脸FacePlay">
</a>
</div>
</td>
<td className='center-column'> <p>${round(token.price, 10000000)}</p> </td>
<td>
<p>{round(token.balance / Math.pow(10, token.decimals), 10000000)}</p>
<p className='token-hold'>${round(token.valueUsd, 10000000)}</p>
</td>
</tr>
))}
</table>
</>
)}
</table>
</div>
</content>
);
}
export default App;关键点解析:
- useState(''): searchValue被初始化为空字符串。这样,在组件首次渲染时,useEffect中的条件if (searchValue)将为假,从而避免了向https://api.multiversx.com/accounts//tokens这样的无效URL发起请求。
-
: 这是在React中渲染组件的正确方式。切勿直接调用InputField()函数,因为那样会绕过React的组件生命周期和状态管理机制。 - useEffect依赖数组[searchValue]: 这是至关重要的。它告诉React,只有当searchValue的值发生变化时,才重新运行useEffect内部的代码。这确保了每次用户提交新的地址时,都会触发一次新的数据请求。
总结与最佳实践
通过上述步骤,我们成功地实现了从子组件InputField向父组件App传递数据,并利用该数据触发API请求。这种模式是React中处理组件间通信的常见且推荐的方式。
核心概念回顾:
- 单向数据流: 数据从父组件流向子组件。子组件通过回调函数通知父组件更新状态。
- 状态提升: 当多个组件需要共享或响应同一个数据时,将该数据提升到它们最近的共同父组件中管理。
- 回调函数作为Props: 父组件定义一个函数,并将其作为prop传递给子组件。子组件在特定事件发生时调用这个函数,将数据传回父组件。
- useState: 用于在函数组件中声明和管理状态。
- useEffect: 用于处理组件的副作用,如数据获取、订阅或手动更改DOM。它的依赖数组决定了何时重新运行副作用。
遵循这些原则,可以构建出结构清晰、可维护性强的React应用。
以上就是React组件间数据传递:从子组件向父组件通信的最佳实践的详细内容,更多请关注其它相关文章!
# 为空
# 南通网站建设首页哪家好
# 东莞seo网络优化服务
# 餐馆如何抖音营销推广
# seo 怎么优化
# 淄博网站推广引流
# 沈阳网络建设网站
# 餐饮企业首页网站建设
# 安庆企业营销推广去哪里
# 网站优化技术宝典
# 小企业优化网站建设建议
# 自定义
# 并将
# 首次
# css
# 一个函数
# 这是
# 是一个
# 复选框
# 从子
# 回调
# 应用开发
# 回调函数
# app
# 浏览器
# json
# js
# react
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
菜鸟取件码是什么怎么查 最全查询渠道汇总
从OpenAI API响应中高效提取生成文本
mc.js免安装版 mc.js一键畅玩入口
怎样更改Windows系统的默认安装路径_避免C盘爆满的终极设置【技巧】
MAC如何将整个网页截长图_MAC使用Safari的导出为PDF或第三方工具
夸克浏览器图书入口 夸克手机浏览器阅读入口
解决 Express.js 中 PUT 请求密码修改失败的路由配置指南
如何创建没有密码的Windows本地账户_跳过微软账户登录的技巧【教程】
poki免费入口快捷访问 poki人气小游戏直接玩站点
Descript怎样用AI剪辑自动去噪_Descript用AI剪辑自动去噪【自动降噪】
零跑汽车11月交付量达70327台 实现连续9个月正增长
《马克思佩恩3》早期版本曝光 UI设计曾多次调整!
HTML5原生日期选择器与jQuery UI:实现日期选择器的联动与程序化控制
邮政快递包裹最新位置 邮政快递实时追踪入口
BetterDiscord插件中安全更新用户简介的实践指南
微博网页版怎么开启两步验证_微博网页版账号安全两步验证设置方法
在J*a中如何开发在线活动报名与管理系统_活动报名管理项目实战解析
Python Socket多播通信中指定源IP地址的实践指南
AO3网页版合集入口 Archive of Our Own同人作品浏览指南
Golang如何安装Swagger工具_GoSwagger文档生成环境
jQuery Mask 插件中实现电话号码固定前导零的教程
LINUX怎么设置定时任务_LINUX crontab配置教程
QQ邮箱登录平台入口 QQ邮箱网页版邮箱官方入口
Win11怎么开启省电模式_Win11电池节电模式自动开启
实现分段式页面滚动导航:CSS与J*aScript教程
免费抖音短视频入口_抖音网页版短视频免费通道
Archive of Our Own官网直达 AO3最新可用地址一览
sublime如何配置Python开发环境_将sublime打造成轻量级Python IDE
微信网页版官方入口直达 微信网页版网页版登录使用方法
一加Ace 6T支持全新明眸护眼:通过了最严苛的护眼小金标认证
J*aScript中高效清空DOM列表元素:解决for循环中断与任务管理问题
漫蛙Manwa2官网入口地址分享 漫蛙漫画PC版永久访问通道
2025AO3夸克浏览器通道_AO3手机HTTPS安全入口分享
cad如何更改注释性对象的比例_cad注释性比例调整方法
解决Bootstrap卡片顶部边距导致背景图下移的问题
文心一言怎样用批量生成做多版文案_文心一言用批量生成做多版文案【批量创作】
Golang切片为何属于引用类型_Golang slice底层结构与引用语义说明
UC浏览器如何安装插件 UC浏览器添加扩展程序详细教程【进阶】
C++ vector二维数组定义_C++ vector of vector用法
iCloud登录入口网页版 苹果iCloud官网登录
Python多版本共存与虚拟环境管理深度指南
铁路12306官网网页端快速入口 铁路12306官方首页登录教程
Linux如何排查内存不足OOME问题_LinuxOOM分析教程
c++如何实现一个简单的软件渲染器_c++从零开始的3D图形学
微博网页版主页入口 微博官方网站免登录访问
韩剧圈正版入口页面_韩剧圈官网登录链接
天眼查怎么看公司融资情况 天眼查企业融资历史查询步骤【攻略】
高德地图怎么看全景照片_高德地图全景照片浏览教程
深入理解rpy2中的类型转换:优化Python对象到R矩阵的映射
Golang如何实现Web接口签名验证_Golang Web接口签名校验开发方法


2025-10-13
浏览次数:次
返回列表
n-side">
<table className="Token-section-output">
{loading ? (
<div>Loading...</div>
) : (
<>
<h1>Tokens</h1>
<table className='Token-section-output' border={0}>
<tr className='token-row-type'>
<th className='token-column'>Name</th>
<th className='center-column'>Price</th>
<th>Hold</th>
</tr>
<tr className="space20"/>
{tokens.map(token => (
<tr className='token-row' key={token.id}>
<td className='token-column'>
@@##@@
<p>{token.name}</p>
<div class="aritcle_card">
<a class="aritcle_card_img" href="/ai/1247">
<img src="https://img.php.cn/upload/ai_manual/000/000/000/175680148166020.png" alt="秀脸FacePlay">
</a>
<div class="aritcle_card_info">
<a href="/ai/1247">秀脸FacePlay</a>
<p>一款集成AI换脸、照片跳舞等多种AI特效玩法的App</p>
<div class="">
<img src="/static/images/card_xiazai.png" alt="秀脸FacePlay">
<span>124</span>
</div>
</div>
<a href="/ai/1247" class="aritcle_card_btn">
<span>查看详情</span>
<img src="/static/images/cardxiayige-3.png" alt="秀脸FacePlay">
</a>
</div>
</td>
<td className='center-column'> <p>${round(token.price, 10000000)}</p> </td>
<td>
<p>{round(token.balance / Math.pow(10, token.decimals), 10000000)}</p>
<p className='token-hold'>${round(token.valueUsd, 10000000)}</p>
</td>
</tr>
))}
</table>
</>
)}
</table>
</div>
</content>
);
}
export default App;