深入理解 React 中的 forwardRef :实现子组件 DOM 控制

引言

在 React 开发中,组件化是核心思想之一。然而,随着项目规模的增长,组件之间的交互和样式管理变得越来越复杂。本文将重点讲解两个在 React 开发中常见的关键知识点:forwardRef 的使用CSS 模块化解决样式冲突(CSS Bug) ,帮助你构建更健壮、可维护的 React 应用。

一、forwardRef:让函数组件也能接收 ref

1.1 什么是 forwardRef

forwardRef 是 React 提供的一个高阶函数,用于ref 从父组件传递给子组件内部的 DOM 元素或子组件

默认情况下,函数组件不能直接接收 ref ,因为它们没有实例。而 forwardRef 的作用就是让函数组件也能"转发" ref 到其内部的 DOM 元素上。

1.2 使用场景

  • 父组件需要访问子组件内部的 DOM 元素(如 <input><canvas>)。
  • 封装组件时,希望外部能控制组件内部的焦点、滚动、动画等行为。
  • 构建第三方组件库时,提供对内部 DOM 的访问能力。

1.3 基本用法

jsx 复制代码
import { 
  useEffect,
  useRef,
  forwardRef
 } from 'react'
import './App.css'

function Guang (props, ref){
  console.log(props,ref);
  return (
    <div>
      <input type="text" ref={ref}/>
    </div>
  )
}

// 高阶组件
const WrapperGuang = forwardRef(Guang); // ref要传递的组件作为参数传递给它

function App() {
  // 父组件 持有ref
  const ref = useRef(null); 
  console.log(ref.current); // null
  
  useEffect(() => {
    ref.current?.focus();
  }, [])
  return (
    <div className='App'>
      {/* 2. 父组件通过ref传递给子组件 */}
     <WrapperGuang title="hello" ref={ref}/>
    </div>
  )
}

export default App

定义了一个组件 Guang,并使用 forwardRef 包裹它,形成 WrapperGuang,然后在 App 父组件中通过 ref 将其指向子组件内部的 <input> 元素。

效果图:

forwardRef 的作用

问题背景:

在 React 中,默认情况下,父组件不能直接访问子组件内部的 DOM 元素或子组件实例 。也就是说,如果在父组件中定义了一个 ref 并传给子组件,它不会自动传递到子组件的 DOM 节点上。

jsx 复制代码
const Child = () => <input />;

const Parent = () => {
  const ref = useRef();
  return <Child ref={ref} />; // ❌ ref 不会自动传给 input
}
解决方案:使用 forwardRef

forwardRef 是一个高阶函数,用于 显式地将父组件传来的 ref 转发给子组件内部的某个 DOM 元素或子组件

✅ 三、代码逐段解析

1. 定义组件 Guang

jsx 复制代码
function Guang(props, ref) {
  console.log(props, ref);
  return (
    <div>
      <input type="text" ref={ref} />
    </div>
  )
}
  • Guang 是一个函数组件。
  • 它接收两个参数:propsref
  • 它将传入的 ref 绑定到了内部的 <input> 元素上。
  • 所以这个组件内部的 <input> 元素就可以被外部通过 ref 控制。

2. 使用 forwardRef 创建高阶组件

jsx 复制代码
const WrapperGuang = forwardRef(Guang);
  • 使用 forwardRef 包装 Guang,使它能够接收来自父组件的 ref
  • 这样父组件就可以通过 ref 来访问子组件中的 <input>

3. 父组件 App

jsx 复制代码
function App() {
  const ref = useRef(null); 
  console.log(ref.current); // null(此时 DOM 尚未渲染)

  useEffect(() => {
    ref.current?.focus(); // 组件挂载后自动聚焦
  }, [])

  return (
    <div className='App'>
      <WrapperGuang title="hello" ref={ref} />
    </div>
  )
}
  • useRef(null) 创建了一个 ref。
  • useEffect 中使用 ref.current?.focus() 实现自动聚焦。

可选链操作符?.)用于访问对象属性或调用方法时,若路径中某部分为nullundefined,则表达式短路返回undefined,避免抛出错误。 如果 ref.currentnull 或者 undefined,那么这个表达式会短路返回 undefined 而不会抛出错误。

  • <WrapperGuang title="hello" ref={ref} />:将 ref 传给了子组件,并最终绑定到 <input> 上。

结尾

通过 forwardRef,我们能够优雅地将父组件的 ref 传递到子组件内部的 DOM 元素上,实现对子组件 DOM 的直接访问与操作。

相关推荐
QQ1__8115175154 小时前
Spring boot名城小区物业管理系统信息管理系统源码-SpringBoot后端+Vue前端+MySQL【可直接运行】
前端·vue.js·spring boot
钛态4 小时前
前端微前端架构:大项目的救命稻草还是自找麻烦?
前端·vue·react·web
一粒黑子4 小时前
【实战解析】阿里开源 PageAgent:纯前端 GUI Agent,一行JS让网页支持自然语言操控
前端·javascript·开源
独角鲸网络安全实验室4 小时前
2026微信小程序抓包全解析:从实操落地到合规风控,解锁前端调试新范式
前端·微信小程序·小程序·抓包·系统代理绕过·https证书严格校验·进程隔离
紫微AI4 小时前
前端文本测量成了卡死一切创新的最后瓶颈,pretext实现突破了
前端·人工智能·typescript
GISer_Jing4 小时前
AI前端(From豆包)
前端·aigc·ai编程
IT枫斗者4 小时前
前端部署后如何判断“页面是不是最新”?一套可落地的版本检测方案(适配 Vite/Vue/React/任意 SPA)
前端·javascript·vue.js·react.js·架构·bug
测试修炼手册4 小时前
[测试技术] 深入理解 JSON Web Token (JWT)
前端·json
AI老李4 小时前
2026 年 Web 前端开发的 8 个趋势!
前端
里欧跑得慢4 小时前
15. Web可访问性最佳实践:让每个用户都能平等访问
前端·css·flutter·web