【React】详解如何获取 DOM 元素

文章目录

    • 一、基础概念
      • 1. 什么是DOM?
      • 2. 为什么需要获取DOM?
    • 二、使用 `ref` 获取DOM元素
      • 1. 基本概念
      • 2. 类组件中的 `ref`
      • 3. 函数组件中的 `ref`
    • 三、 `ref` 的进阶用法
      • 1. 动态设置 `ref`
      • 2. `ref` 与函数组件的结合
    • 四、处理特殊情况
      • 1. 多个 `ref` 的处理
      • 2. `ref` 与条件渲染

在React中,操作DOM是一个常见但经常让人感到困惑的任务。虽然React鼓励使用虚拟DOM来优化性能和简化开发,但有时你仍然需要直接访问实际的DOM元素来完成特定任务。本文将详细介绍在React中获取DOM元素的各种方法,包括基本的 ref 使用、函数组件中的 ref,以及如何在React Hooks中利用 useRef。通过本文,你将全面了解如何在React中高效、安全地获取和操作DOM元素。

一、基础概念

1. 什么是DOM?

DOM(文档对象模型)是HTML和XML文档的编程接口。它将文档表示为一个由节点构成的树形结构,每个节点代表文档中的一个部分,比如元素、属性或文本。在React中,虽然大多数操作通过虚拟DOM进行,但有时你仍然需要直接操作实际的DOM节点。

2. 为什么需要获取DOM?

在React中,通常推荐使用虚拟DOM来处理用户界面更新,但某些情况下你可能需要直接访问DOM节点。例如:

  • 当与第三方库集成时,需要直接操作DOM。
  • 在进行动画时,可能需要访问实际的DOM节点。
  • 实现自定义UI组件时,有时需要直接操作DOM。

二、使用 ref 获取DOM元素

1. 基本概念

在React中,ref 是一个特殊的属性,用于引用DOM节点或类组件实例。通过 ref,你可以在函数组件或类组件中访问实际的DOM元素。

2. 类组件中的 ref

在类组件中,可以使用 React.createRef() 创建 ref,并将其附加到一个DOM元素上:

import React, { Component } from 'react';class MyComponent extends Component {constructor(props) {super(props);this.myRef = React.createRef();}componentDidMount() {// 通过 this.myRef.current 访问实际的DOM节点console.log(this.myRef.current);}render() {return (<div><input ref={this.myRef} type="text" /></div>);}
}export default MyComponent;

在上面的示例中:

  • React.createRef() 创建了一个 ref 对象。
  • this.myRef 被附加到 <input> 元素上。
  • componentDidMount 生命周期方法中,你可以通过 this.myRef.current 访问到实际的DOM节点。

3. 函数组件中的 ref

在函数组件中,可以使用 useRef Hook 来创建 ref

import React, { useRef, useEffect } from 'react';function MyComponent() {const inputRef = useRef(null);useEffect(() => {// 通过 inputRef.current 访问实际的DOM节点console.log(inputRef.current);}, []);return (<div><input ref={inputRef} type="text" /></div>);
}export default MyComponent;

在这个示例中:

  • useRef 创建了一个 ref 对象,初始值为 null
  • inputRef 被附加到 <input> 元素上。
  • useEffect Hook 在组件挂载后运行,可以通过 inputRef.current 访问到实际的DOM节点。

三、 ref 的进阶用法

1. 动态设置 ref

在某些情况下,你可能需要根据条件动态设置 ref

import React, { useRef, useEffect } from 'react';function MyComponent({ isFocused }) {const inputRef = useRef(null);useEffect(() => {if (isFocused && inputRef.current) {inputRef.current.focus();}}, [isFocused]);return (<div><input ref={inputRef} type="text" /></div>);
}export default MyComponent;

在这个示例中:

  • useEffect 会在 isFocused 变化时执行。
  • 如果 isFocusedtrue,则会使 inputRef.current 获取焦点。

2. ref 与函数组件的结合

你也可以将 ref 传递给子组件:

import React, { forwardRef, useRef, useImperativeHandle } from 'react';const ChildComponent = forwardRef((props, ref) => {useImperativeHandle(ref, () => ({alertMessage() {alert('Hello from ChildComponent!');}}));return <div>Child Component</div>;
});function ParentComponent() {const childRef = useRef(null);const handleClick = () => {if (childRef.current) {childRef.current.alertMessage();}};return (<div><ChildComponent ref={childRef} /><button onClick={handleClick}>Call Child Component</button></div>);
}export default ParentComponent;

在这个示例中:

  • ChildComponent 使用 forwardRefuseImperativeHandle 定义了可供父组件调用的 ref 方法。
  • ParentComponent 可以通过 childRef.current.alertMessage() 调用子组件中的方法。

四、处理特殊情况

1. 多个 ref 的处理

当需要在同一个组件中管理多个 ref 时,可以使用多个 useRef 或创建一个对象来存储多个 ref

import React, { useRef, useEffect } from 'react';function MyComponent() {const refs = {inputRef1: useRef(null),inputRef2: useRef(null)};useEffect(() => {console.log(refs.inputRef1.current);console.log(refs.inputRef2.current);}, []);return (<div><input ref={refs.inputRef1} type="text" placeholder="Input 1" /><input ref={refs.inputRef2} type="text" placeholder="Input 2" /></div>);
}export default MyComponent;

在这个示例中:

  • refs 对象存储了多个 ref
  • 可以通过 refs.inputRef1.currentrefs.inputRef2.current 访问不同的DOM节点。

2. ref 与条件渲染

在条件渲染的情况下,可能需要处理 ref 的存在性:

import React, { useRef, useEffect, useState } from 'react';function MyComponent() {const [isVisible, setIsVisible] = useState(true);const inputRef = useRef(null);useEffect(() => {if (isVisible && inputRef.current) {console.log('Input is visible:', inputRef.current);}}, [isVisible]);return (<div>{isVisible && <input ref={inputRef} type="text" />}<button onClick={() => setIsVisible(!isVisible)}>Toggle Input</button></div>);
}export default MyComponent;

在这个示例中:

  • 使用 useState 控制 <input> 的显示和隐藏。
  • useEffect 检查 isVisibleinputRef.current 的状态来确保 ref 只在 <input> 可见时有效。

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

大数据-49 Redis 缓存问题中 穿透、雪崩、击穿、数据不一致、HotKey、BigKey

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff08;已更完&#xff09;HDFS&#xff08;已更完&#xff09;MapReduce&#xff08;已更完&am…

nginx目录列表美化—rpm安装

目录美化 1. 下载NGINX2. 下载美化工具3. 配置模块4. 主题下载5. 配置文件编写6. 其它问题 1. 下载NGINX RHEL系列的yum源 使用yum源安装如果不能指定版本&#xff0c;请点击跳转nginx的仓库 nginx-stable] namenginx stable repo baseurlhttp://nginx.org/packages/centos/$…

【H.264】H.264详解(二)—— H264视频码流解析示例源码

文章目录 一、前言二、示例源码【1】目录结构【2】Makefile源码【3】h264parser.c源码【4】编译运行【5】源码下载地址 声明&#xff1a;此篇示例源码非原创&#xff0c;原作者雷霄骅。雷霄骅&#xff0c;中国传媒大学通信与信息系统专业博士生&#xff0c;在此向雷霄骅雷神致敬…

【Python机器学习】朴素贝叶斯——条件概率

条件概率 假设现在有一个装了7块石头的罐子&#xff08;3块灰色&#xff0c;4块黑色&#xff09;&#xff0c;如果从中随机取出一块&#xff0c;灰色的可能性就是3/7&#xff0c;黑色的可能性是4/7。我们使用p(gray)来表示取到灰色石头的概率&#xff0c;其概率值可以通过灰色…

cocos creator 3学习记录01——如何替换图片

一、动态加载本地图片 1、通过将图片关联到CCClass属性上来进行代码切换。 1、这种方法&#xff0c;需要提前在脚本文件中声明好代表图片的CCClass属性。 2、然后拖动图片资源&#xff0c;到脚本内声明好的属性上以进行关联。 3、然后通过程序&#xff0c;来进行切换展示。…

unity2D游戏开发01项目搭建

1新建项目 选择2d模板,设置项目名称和存储位置 在Hierarchy面板右击&#xff0c;create Empty 添加组件 在Project视图中右键新建文件夹 将图片资源拖进来&#xff08;图片资源在我的下载里面&#xff09; 点击Player 修改属性&#xff0c;修好如下 点击Sprite Editor 选择第二…

Hadoop3:HDFS的客户端工具Big Data Tools(IDEA版本)

1、安装插件 在Plugins里搜索Big Data Tools 安装完成后&#xff0c;重启IDEA 2、配置Windows环境 主要是配置Hadoop环境&#xff0c;否则无法通过插件远程连接HDFS 1、解压hadoop安装包 2、进入hadoop的bin目录 放入图中标红的两个文件 3、配置hadoop环境变量 新建HAD…

freertos的学习cubemx版

HAL 库的freertos 1 实时 2 任务->线程 3 移植 CMSIS_V2 V1版本 NVIC配置全部是抢占优先级 第四组 抢占级别有 0-15 编码规则&#xff0c; 变量名 &#xff1a;类型前缀&#xff0c; c - char S - int16_t L - int32_t U - unsigned Uc - uint8_t Us - uint…

【游戏制作】使用Python创建一个完整的2048游戏项目

目录 项目运行展示 项目概述 项目目标 项目结构 安装依赖 代码实现 1. 导入库 2. 创建 Game2048 类 3. 设置UI界面 4. 加载二维码图片 5. 创建菜单 6. 游戏逻辑和功能 7. 运行应用 总结 创建一个完整的2048游戏项目 项目运行展示 项目概述 在这个项目中&#xff…

19 Python常用内置函数——range()

range() 是 Python 开发中非常常用的一个内置函数。该函数返回具有惰性求值特点的 range 对象&#xff0c;其中包含左闭右开区间 [start, end) 内以 step 为步长的整数。 参数 start 默认为 0&#xff0c;step 默认为 1。 print(range(5)) print(list(range(5))) print(list(r…

Zenario CMS 9.2 文件上传漏洞(CVE-2022-23043)

前言 CVE-2022-23043 是一个影响 Zenario CMS 9.2 的严重漏洞。该漏洞允许经过身份验证的管理员用户绕过文件上传限制。具体来说&#xff0c;管理员可以通过创建一个新的带有 ".phar" 扩展名的“文件/MIME 类型”&#xff0c;然后上传一个恶意文件。在上传过程中&am…

Java从入门到精通 (十一) ~ 操作系统、进程和线程

无论做什么&#xff0c;请记住都是为你自己而做&#xff0c;这样就毫无怨言&#xff01;今天&#xff0c;我为自己而活&#xff01;今天&#xff0c;又是美丽的一天&#xff01;早安&#xff0c;朋友&#xff01; 目录 前言 一、操作系统 1. 概念 2. 操作系统的基本功能 3…

FPGA开发——LED流水灯实现先从左往右流水,再从右往左流水

一、概述 我们在设计完一个方向的流水灯的设计时&#xff0c;总是会想实现让流水灯倒着流水回去的设计&#xff0c;这里我也是一样&#xff0c;实现这种设计的方法有很多种&#xff0c;其中就有直接使用case语句将所有可能包含进去编写&#xff0c;这种设计方法是最简单的&…

高翔【自动驾驶与机器人中的SLAM技术】学习笔记(四)高斯牛顿法详解

一、高斯牛顿法详解 拓展阅读&#xff1a;高斯牛顿法详解_gauss-newton算法步骤-CSDN博客 1、梯度下降法 ​ ​ ​ 无论一阶泰勒展开&#xff0c;还是二阶泰勒展开都是关于增量​的方程。 2、牛顿法 ​ 这个自变量增量都是可求的。但是二阶求解复杂。因此为了简化有了下…

windows下运行sh文件

1、打开git bash 2、进入sh文件所在文件夹&#xff0c;使用sh xx.sh运行

@RequiredArgsConstructor详解

RequiredArgsConstructor详解 一、什么是RequiredArgsConstructor? RequiredArgsConstructor是Lombok的一个注解&#xff0c;简化了我们对Autowired书写&#xff0c;我们在写Controller层或者Service层的时候&#xff0c;总是需要注入很多mapper接口或者service接口&#xf…

初识Play Framework框架和第一个Java play web项目的创建

文章目录 初识Play Framework框架和第一个Java play web项目的创建一、简介特点架构开发流程示例代码总结 二、创建第一个Java play web项目1、下载play框架&#xff0c;配置系统环境变量(jdk的安装就不再说了) 2、检查play的版本和创建第一个play项目3、将项目通过idea或eclip…

【NoSQL数据库】Redis知识小册

一、缓存穿透 缓存穿透是先查Redis&#xff0c;发现缓存中没有数据&#xff0c;再查数据库。然而&#xff0c;如果查询的数据在数据库中也不存在&#xff0c;那么每次查询都会绕过缓存&#xff0c;直接落到数据库上。 解决方案一、缓存空数据 查询Redis缓存&#xff1a;首先查…

Java泛型的介绍和基本使用

什么是泛型 ​ 泛型就是将类型参数化&#xff0c;比如定义了一个栈&#xff0c;你必须在定义之前声明这个栈中存放的数据的类型&#xff0c;是int也好是double或者其他的引用数据类型也好&#xff0c;定义好了之后这个栈就无法用来存放其他类型的数据。如果这时候我们想要使用这…

Codeforces 903 div3 A-F

A 题目分析 数据范围很小&#xff0c;暴力枚举即可&#xff0c;然后给字符串x的长度设置一个上限&#xff0c;我设了50&#xff0c;因为n*m<25&#xff0c;多一倍够用了 C代码 #include<iostream> using namespace std; void solve(){int n,m;string x,s;cin>>…