如何使用React,透传各类组件能力/属性?

在23年的时候,我主要使用的框架还是Vue,当时写了一篇“如何二次封装一个Vue3组件库?”的文章,里面涉及了一些如何使用Vue透传组件能力的方法。在我24年接触React之后,我发现这种扩展组件能力的方式有一个专门的术语:高阶组件(HOC)。但在Vue开发中,这个词很少听到。

这篇文章中会描述使用React透传组件各类能力的方式。这些透传方式经常在高阶组件中使用,但并不只有高阶组件会用到它们。React有类式组件和函数式组件两种,我们会分别介绍。

问题描述

首先我们列举下简单的场景,说明我们为什么需要透传组件能力。这里以函数式组件为例。

// 问题示例,这段代码是不正确的
import { useRef } from "react";function FunComp() {return <input />;
}function FunComp2() {return <div><input /></div>;
}function App() {const refFun = useRef(null);const handleClick = () => {console.log("click");};const handleClickFocus = () => {if (refFun.current) refFun.current?.focus();};return (<div><FunCompref={refFun}style={{ background: "red" }}onClick={handleClick}/><div onClick={handleClickFocus}>点我聚焦</div></div>);
}

假设我们想创建一个自定义组件(FunComp),里面封装了另外一个组件(例如这里的input),希望使用这个自定义组件增强原组件的能力,或者预先设定一些样式等等,自定义组件可能直接返回该组件,也可能被处理过(例如FunComp2被div包裹)。

虽然原组件被封装了,但是还希望原组件的能力直接被透传给自定义组件。例如我们在自定义组件上操作props, 事件,ref等,希望就像操作原组件一样。在Vue3中,很多能力可以直接使用Attributes继承特性,但是React却没有,需要我们自己实现。

函数式组件-透传Props和事件

在函数式组件中,prop实际上就是组件的入参,且所有prop是被包含在同一个参数中的,因此很容易透传给子组件。且事件本身实际上也是prop,可以一并透传。

function FunComp(props) {return <input {...props} />;
}function App() {const handleClick = () => {console.log("click");};return (<div><FunComp style={{ background: "red" }} onClick={handleClick} /></div>);
}

使用{...props}可以实现透传Props和事件。上述例子中,子组件可以接收到样式属性和事件。

函数式组件-透传子节点

React中有一个特殊的属性children,表示父组件中包含的子节点。这也是需要透传的。

直接渲染children属性

function FunComp(props) {return <div>{props.children}</div>;
}function App() {return (<div><FunComp>子节点</FunComp><FunComp /></div>);
}

可以看到,直接渲染props.children,即可透传子节点。即使没有子节点,这种透传也是没问题的。如果子组件本身已经透传了props,透传的对象又是

使用透传Props实现透传子节点

既然children也是Props之一,那么直接使用透传Props的方法是否可以呢? 我们试一下。

function FunComp(props) {return <div>{props.children}</div>;
}function FunComp1(props) {return <div {...props} />;
}function FunComp2(props) {return <FunComp {...props} />;
}function App() {return (<div><FunComp1>子节点</FunComp1><FunComp1 children="子节点" /><FunComp2>子节点</FunComp2><FunComp2 children="子节点" /></div>);
}/* 页面效果
子节点
子节点
左 子节点 右
左 子节点 右
*/

FunComp1包含的子组件是一个非自定义组件div,FunComp2包含的时自定义组件FunComp,可以看到我们使用{...props}进行Props透传,children实际上都被成功渲染了,甚至对父组件直接设置children属性也可以。

冲突场景

既然Props透传即可实现,那我们为什么还要强调一遍直接渲染props.children呢,因为有时候子组件不只渲染children,还有其它内容。如果Props和直接设置的子节点冲突,那么还是直接设置的子节点优先级更高。

function FunComp(props) {return <div {...props}>{props.children}</div>;
}function App() {return (<div><FunComp>子节点</FunComp><FunComp children="子节点" /></div>);
}/* 页面效果
左 子节点 右
左 子节点 右
*/

我们同时透传了Props,也直接设置了子节点(其中包含其它内容),最后直接设置的子元素生效了。

函数式组件-透传ref

使用ref可以操作访问DOM节点,获取DOM元素上的属性或者方法。ref也是可以透传的。

透传全部属性

import { forwardRef, useRef } from "react";const FunComp = forwardRef(function (props, ref) {return <input ref={ref} />;
});function App() {const inputRef = useRef(null);function handleClick() {console.log(inputRef.current?.style);inputRef.current?.focus();}return (<div><FunComp ref={inputRef} /><div onClick={handleClick}>点击聚焦</div></div>);
}

使用forwardRef,可以透传ref属性。我们尝试了聚焦输入框,以及console输出style属性,都是正常生效的。

仅暴露部分属性

有时候我们不想暴露全部属性,仅希望暴露我们希望用户使用的部分属性,使用useImperativeHandle可以做到。

import { forwardRef, useRef, useImperativeHandle } from "react";const FunComp = forwardRef(function (props, ref) {const inputRef = useRef(null);useImperativeHandle(ref, () => {return {focus() {inputRef.current?.focus();},};});return <input ref={inputRef} />;
});function App() {const inputRef = useRef(null);function handleClick() {// 无法输出console.log(inputRef.current?.style);inputRef.current?.focus();}return (<div><FunComp ref={inputRef} /><div onClick={handleClick}>点击聚焦</div></div>);
}

我们仅向外层的ref暴露了focus,因此外层组件focus可以正常调用,但是却拿不到style属性了。使用这种形式还可以对方法进行额外的包装,或者创建一些新的ref方法。

在React19中,不再需要forwardRef了,ref直接作为一个prop属性访问。可以看最后的参考文档。

类式组件-透传Props和事件

类式组件是另一种创建React组件的方法,被React标记为过时的API,但是在老代码中还经常被使用到。我们先来看一下,在类式组件中,如何Props和事件。

import { Component } from "react";
class ClassComp extends Component {render() {return <div {...this.props}>你好</div>;}
}class App extends Component {render() {const handleClick = () => {console.log("click");};return <ClassComp style={{ background: "red" }} onClick={handleClick} />;}
}

通过上述代码可以看到,在类式组件中透传Props和事件与函数式组件一致,使用{...props}可以实现透传Props和事件。

类式组件-透传子节点

来看看类式组件是如何透传子节点的。

直接渲染children属性

import { Component } from "react";
class ClassComp extends Component {render() {return <div>{this.props.children}</div>;}
}class App extends Component {render() {return (<div><ClassComp>子节点</ClassComp><ClassComp /></div>);}
}

代码依然与类式组件基本一致,直接渲染{this.props.children}即可。

使用透传Props实现透传子节点

上一节讲到的透传Props,同样可以实现透传子节点。

import { Component } from "react";
class ClassComp extends Component {render() {return <div>{this.props.children}</div>;}
}
class ClassComp1 extends Component {render() {return <div {...this.props} />;}
}
class ClassComp2 extends Component {render() {return <ClassComp {...this.props} />;}
}class App extends Component {render() {return (<div><ClassComp1>子节点</ClassComp1><ClassComp1 children="子节点" /><ClassComp2>子节点</ClassComp2><ClassComp2 children="子节点" /></div>);}
}/* 页面效果
子节点
子节点
左 子节点 右
左 子节点 右
*/

与函数式组件一致,Props透传时也会透传children,甚至对父组件直接设置children属性也可以透传。至于Props和直接设置的子节点冲突的场景也与函数式组件一致,这里就不举例了。

类式组件-透传ref

类式组件透传Ref的形式就与函数式组件不同了。具体类式组件有不同的实现方式,我们分别介绍下:

暴露部分属性

import { Component, createRef } from "react";class ClassComp extends Component {inputRef = createRef();focus() {this.inputRef.current?.focus();}render() {return <input ref={this.inputRef} />;}
}class App extends Component {classRef = createRef();handleClick() {console.log(this.classRef.current);this.classRef.current?.focus();}render() {return (<div><ClassComp ref={this.classRef} /><div onClick={() => this.handleClick()}>点击聚焦</div></div>);}
}

通过代码可以看到,在类式组件中,不需要通过forwardRef等方法就可以使用ref访问子组件,且能执行子组件类中的方法。所以我们只要把需要暴露的内容包装成一个方法,那么就可以让父组件获取到。

在这里插入图片描述

通过输出的图可以看到,不仅能拿到方法,还能拿到属性和其它很多东西。

拿到子组件内部的ref

既然可以拿到子组件类中和属性也能拿到。那么父组件可以直接拿到子组件内部的ref属性inputRef,父组件可以直接拿到它来执行内部的方法。

import { Component, createRef } from "react";class ClassComp extends Component {inputRef = createRef();render() {return <input ref={this.inputRef} />;}
}class App extends Component {classRef = createRef();handleClick() {this.classRef.current?.inputRef?.current?.focus();}render() {return (<div><ClassComp ref={this.classRef} /><div onClick={() => this.handleClick()}>点击聚焦</div></div>);}
}

通过代码可以看到,子组件不需要透出方法了,父组件直接拿到子组件的inputRef,想执行什么就执行什么,做到了真正的“透传”。绑定ref还有另一种方式,这里也介绍一下:

import { Component, createRef } from "react";class ClassComp extends Component {render() {return <input ref="inputRef" />;}
}class App extends Component {classRef = createRef();handleClick() {this.classRef.current?.refs?.inputRef?.focus();}render() {return (<div><ClassComp ref={this.classRef} /><div onClick={() => this.handleClick()}>点击聚焦</div></div>);}
}

ref属性的值可以直接是一个字符串,通过this.refs可以拿到使用字符串形式绑定的ref。

总结

函数式组件与类式组件在Props和事件透传的方式基本一致,但是ref透传的区别较大。直接对比的话,好像类式组件的透传能力更强一些,但是它把组件内部所有内容全暴露在外,违反了封装的原则,子组件内部的改动很容易影响父组件,不是一个好的设计。

在React19版本中,ref属性也变成了prop,仅通过透传Props,就能实现透传组件大部分能力了。

参考

  • 如何二次封装一个Vue3组件库?
    https://jzplp.github.io/2023/component-lib.html
  • Vue3 透传Attributes
    https://cn.vuejs.org/guide/components/attrs
  • React 使用ref操作DOM
    https://zh-hans.react.dev/learn/manipulating-the-dom-with-refs
  • React omponent
    https://zh-hans.react.dev/reference/react/Component
  • ref用法
    https://blog.csdn.net/qq_47305413/article/details/136059266
  • React v19 ref作为一个属性
    https://zh-hans.react.dev/blog/2024/12/05/react-19#ref-as-a-prop

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

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

相关文章

109.【C语言】数据结构之求二叉树的高度

目录 1.知识回顾&#xff1a;高度&#xff08;也称深度&#xff09; 2.分析 设计代码框架 返回左右子树高度较大的那个的写法一:if语句 返回左右子树高度较大的那个的写法二:三目操作符 3.代码 4.反思 问题 出问题的代码 改进后的代码 执行结果 1.知识回顾&#xf…

通过百度api处理交通数据

通过百度api处理交通数据 1、读取excel获取道路数据 //道路名称Data EqualsAndHashCode public class RoadName {ExcelProperty("Name")private String name; }/*** 获取excel中的道路名称*/private static List<String> getRoadName() {// 定义文件路径&…

分析排名靠前的一些自媒体平台,如何运用这些平台?

众所周知&#xff0c;现在做网站越来越难了&#xff0c;主要的原因还是因为流量红利时代过去了。并且搜索引擎都在给自己的平台做闭环改造。搜索引擎的流量扶持太低了。如百度投资知乎&#xff0c;给知乎带来很多流量扶持&#xff0c;也为自身内容不足做一个填补。 而我们站长…

2024大模型在软件开发中的具体应用有哪些?(附实践资料合集)

大模型在软件开发中的具体应用非常广泛&#xff0c;以下是一些主要的应用领域&#xff1a; 自动化代码生成与智能编程助手&#xff1a; AI大模型能够根据开发者的自然语言描述自动生成代码&#xff0c;减少手动编写代码的工作量。例如&#xff0c;GitHub Copilot工具就是利用AI…

webpack的说明

介绍 因为不确定打出的前端包所访问的后端IP&#xff0c;需要对项目中IP配置文件单独拿出来&#xff0c;方便运维部署的时候对IP做修改。 因此&#xff0c;需要用webpack单独打包指定文件。 CommonsChunkPlugin module.exports {entry: {app: APP_FILE // 入口文件},outpu…

HTML 画布:创意与技术的融合

HTML 画布:创意与技术的融合 HTML 画布(<canvas>)元素是现代网页设计中的一个强大工具,它为开发者提供了一个空白画布,可以在上面通过JavaScript绘制图形、图像和动画。这种技术不仅为网页增添了视觉吸引力,还极大地丰富了用户的交互体验。本文将深入探讨HTML画布…

Ubuntu网络配置(桥接模式, nat模式, host主机模式)

windows上安装了vmware虚拟机&#xff0c; vmware虚拟机上运行着ubuntu系统。windows与虚拟机可以通过三种方式进行通信。分别是桥接模式&#xff1b;nat模式&#xff1b;host模式 一、桥接模式 所谓桥接模式&#xff0c;也就是虚拟机与宿主机处于同一个网段&#xff0c; 宿主机…

【SQL】王二的100道SQL刷题进阶之路

持续更新&#xff0c;建议关注收藏&#xff01; SQL进阶看这一篇就够了&#xff01; 目录 1-datediff2-生成排序序号3-having注意4-procedure declare5-弯弯绕绕 1-datediff select id,datediff(end_date, start_date) as diff from Tasks order by diff desc limit 3;dated…

3.系统学习-熵与决策树

熵与决策树 前言1.从数学开始信息量(Information Content / Shannon information)信息熵(Information Entropy)条件熵信息增益 决策树认识2.基于信息增益的ID3决策树3.C4.5决策树算法C4.5决策树算法的介绍决策树C4.5算法的不足与思考 4. CART 树基尼指数&#xff08;基尼不纯度…

FLV视频封装格式详解

目录(?)[-] OverviewFile Structure The FLV headerThe FLV File BodyFLV Tag Definition FLVTAGAudio TagsVideo TagsSCRIPTDATA onMetaDatakeyframes Overview Flash Video(简称FLV),是一种流行的网络格式。目前国内外大部分视频分享网站都是采用的这种格式. File Structure…

Text2Reward学习笔记

1. 提示词 请问&#xff0c;“glew”是一个RL工程师常用的工具库吗&#xff1f;2. 环境配置 2.1 安装 PyTorch-1.13.1 pip install torch1.13.1cu116 torchvision0.14.1cu116 \ torchaudio0.13.1 --extra-index-url https://download.pytorch.org/whl/cu1161.2 安装工具库 …

SpringBoot + HttpSession 自定义生成sessionId

SpringBoot HttpSession 自定义生成sessionId 业务场景实现方案 业务场景 最近在做用户登录过程中&#xff0c;由于默认ID是通过UUID创建的&#xff0c;缺乏足够的安全性&#xff0c;决定要自定义生成 sessionId。 实现方案 正常的获取session方法如下&#xff1a; HttpSe…

破解海外业务困局:新加坡服务器托管与跨境组网策略

在当今全球化商业蓬勃发展的浪潮之下&#xff0c;众多企业将目光投向海外市场&#xff0c;力求拓展业务版图、抢占发展先机。而新加坡&#xff0c;凭借其卓越的地理位置、强劲的经济发展态势以及高度国际化的营商环境&#xff0c;已然成为企业海外布局的热门之选。此时&#xf…

CMS(Concurrent Mark Sweep)垃圾回收器的具体流程

引言 CMS&#xff08;Concurrent Mark Sweep&#xff09;收集器是Java虚拟机中的一款并发收集器&#xff0c;其设计目标是最小化停顿时间&#xff0c;非常适合于对响应时间敏感的应用。与传统的串行或并行收集器不同&#xff0c;CMS能够尽可能地让垃圾收集线程与用户线程同时运…

数学课程评价系统:客户服务与教学支持

2.1 SSM框架介绍 本课题程序开发使用到的框架技术&#xff0c;英文名称缩写是SSM&#xff0c;在JavaWeb开发中使用的流行框架有SSH、SSM、SpringMVC等&#xff0c;作为一个课题程序采用SSH框架也可以&#xff0c;SSM框架也可以&#xff0c;SpringMVC也可以。SSH框架是属于重量级…

攻防世界web第三题file_include

<?php highlight_file(__FILE__);include("./check.php");if(isset($_GET[filename])){$filename $_GET[filename];include($filename);} ?>惯例&#xff1a; 代码审查&#xff1a; 1.可以看到include(“./check.php”);猜测是同级目录下有一个check.php文…

C++设计模式:解释器模式(简单的数学表达式解析器)

什么是解释器模式&#xff1f; 解释器模式是一种行为型设计模式&#xff0c;用于为特定的语言定义一个解释器&#xff0c;解释并执行语言中的句子。它主要用于构建一个简单的语法解释器&#xff0c;将特定的业务逻辑转化为可理解的语言表达&#xff0c;并对这些表达式进行求值…

【深度学习环境】NVIDIA Driver、Cuda和Pytorch(centos9机器,要用到显示器)

文章目录 一 、Anaconda install二、 NIVIDIA driver install三、 Cuda install四、Pytorch install 一 、Anaconda install Step 1 Go to the official website: https://www.anaconda.com/download Input your email and submit. Step 2 Select your version, and click i…

寻找适合小户型的开源知识库open source knowledge base之路

寻找一个开源的知识库&#xff0c;为了把以前花很多时间收集的信息或是项目/课程资料放到一个容易归类和管理的私有自主系统中&#xff0c;以便更容易查阅&#xff0c;花更少时间收集、对比版本及分享等一系列管理工作&#xff0c;同时确保在需要时可以相对快速找到有用的资料&…

C语言勘破之路-最终篇 —— 预处理(上)

人无完人&#xff0c;持之以恒&#xff0c;方能见真我&#xff01;&#xff01;&#xff01; 共同进步&#xff01;&#xff01; 文章目录 一、预定义符号二、#define定义常量三.、#define定义宏四、带有副作用的宏参数五、宏替换的规则六、宏和函数的对比1.宏的优势2.函数的优…