当前位置: 首页 > article >正文

React与原生事件:核心差异与性能对比解析

💝💝💝欢迎莅临我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。
推荐:「stormsha的主页」👈,「stormsha的知识库」👈持续学习,不断总结,共同进步,为了踏实,做好当下事儿~
非常期待和您一起在这个小小的网络世界里共同探索、学习和成长。💝💝💝 ✨✨ 欢迎订阅本专栏 ✨✨

在这里插入图片描述

💖The Start💖点点关注,收藏不迷路💖

📒文章目录

    • 1. 事件系统基础
      • 1.1 原生DOM事件的工作原理
      • 1.2 React合成事件(SyntheticEvent)概览
    • 2. 核心差异对比
      • 2.1 事件绑定语法
      • 2.2 事件传播控制
      • 2.3 事件委托实现
    • 3. 高级特性与陷阱
      • 3.1 React事件池的注意事项
      • 3.2 混合使用时的冲突
      • 3.3 自定义事件处理
    • 4. 性能优化实践
      • 4.1 避免内联函数
      • 4.2 大规模列表优化
      • 4.3 被动事件监听器
    • 5. 总结


外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

1. 事件系统基础

1.1 原生DOM事件的工作原理

原生DOM事件遵循W3C标准的事件流模型:

document.getElementById('parent').addEventListener('click', () => {console.log('捕获阶段');
}, true); // 第三个参数为true表示捕获阶段document.getElementById('child').addEventListener('click', (event) => {console.log('目标阶段');event.stopPropagation(); // 阻止事件冒泡
});document.getElementById('parent').addEventListener('click', () => {console.log('冒泡阶段');
});
  • 事件对象包含关键属性:
    • target: 触发事件的原始元素
    • currentTarget: 当前处理事件的元素
    • stopPropagation(): 停止事件传播
  • 绑定方式差异
    <!-- 内联方式(不推荐) -->
    <button onclick="handleClick()">Click</button><!-- 标准方式 -->
    <button id="btn">Click</button>
    <script>document.getElementById('btn').addEventListener('click', handleClick);
    </script>
    

1.2 React合成事件(SyntheticEvent)概览

React事件系统的核心设计:

class MyComponent extends React.Component {handleClick = (e) => {console.log(e.nativeEvent); // 访问底层原生事件e.persist(); // 从事件池中移除该事件对象}render() {return <button onClick={this.handleClick}>Click</button>;}
}

关键特性:

  • 跨浏览器兼容:统一了IE和现代浏览器的事件差异
  • 事件池机制:默认会回收事件对象,异步访问需调用e.persist()
  • 自动绑定this:类组件中无需手动绑定事件处理函数的this指向

2. 核心差异对比

2.1 事件绑定语法

特性原生DOMReact
事件名onclickonClick
事件值字符串函数引用
动态绑定需要手动removeEventListener自动处理绑定/解绑

错误示例:

// ❌ 错误:传递的是函数调用结果而非函数引用
<button onClick={handleClick()}>Click</button>// ✅ 正确
<button onClick={handleClick}>Click</button>

2.2 事件传播控制

捕获阶段处理差异:

// 原生DOM可监听捕获阶段
div.addEventListener('click', handler, true);// React需使用特殊属性名
<div onClickCapture={handleCapture}>...</div>

2.3 事件委托实现

原生实现方案:

document.getElementById('list').addEventListener('click', (e) => {if (e.target.tagName === 'LI') {console.log('List item clicked:', e.target.dataset.id);}
});

React的自动委托机制:

  • 所有事件默认委托到document节点
  • 根据组件树结构自动维护事件映射
  • 事件处理器会收到包含组件层级信息的合成事件

3. 高级特性与陷阱

3.1 React事件池的注意事项

典型异步访问问题:

function handleClick(e) {setTimeout(() => {// ❌ 报错:事件属性已被回收console.log(e.clientX); // ✅ 解决方案e.persist();console.log(e.clientX);}, 100);
}

3.2 混合使用时的冲突

安全整合第三方库的模式:

useEffect(() => {const handleExternalEvent = () => {...};// 挂载时绑定externalLib.on('event', handleExternalEvent);// 卸载时清理return () => externalLib.off('event', handleExternalEvent);
}, []);

3.3 自定义事件处理

React组件间通信方案:

// 父组件
<Child onCustomEvent={data => console.log(data)} />// 子组件
<button onClick={() => props.onCustomEvent(payload)}>Trigger Event
</button>

4. 性能优化实践

4.1 避免内联函数

优化方案对比:

// ❌ 每次渲染创建新函数
<button onClick={() => setCount(count + 1)}>+</button>// ✅ 类组件方案
class Counter extends React.Component {handleClick = () => {this.setState(prev => ({ count: prev.count + 1 }));}render() {return <button onClick={this.handleClick}>+</button>;}
}// ✅ 函数组件方案
function Counter() {const handleClick = useCallback(() => {setCount(prev => prev + 1);}, []);return <button onClick={handleClick}>+</button>;
}

4.2 大规模列表优化

虚拟滚动实现要点:

<FixedSizeListheight={400}itemCount={1000}itemSize={50}
>{({ index, style }) => (<div style={style} onClick={handleClick}>Item {index}</div>)}
</FixedSizeList>

4.3 被动事件监听器

滚动性能优化方案:

useEffect(() => {const opts = { passive: true };window.addEventListener('scroll', handleScroll, opts);return () => window.removeEventListener('scroll', handleScroll, opts);
}, []);

5. 总结

关键差异总结表:

维度原生事件React合成事件
事件命名全小写驼峰命名
事件绑定命令式API声明式JSX
事件对象原生EventSyntheticEvent
事件委托需手动实现自动全局委托
性能优化需手动管理监听器自动处理大部分场景

选型建议:

  • 使用原生事件的场景:
    • 需要精确控制事件捕获阶段
    • 需要处理React未封装的特殊事件(如resizeObserver)
    • 性能敏感的底层交互(如游戏循环)

最佳实践:

  1. 优先使用React的事件系统
  2. 避免在render中创建新的事件处理器
  3. 及时清理手动添加的原生事件监听器
  4. 善用事件委托处理动态内容

🔥🔥🔥道阻且长,行则将至,让我们一起加油吧!🌙🌙🌙

💖The Start💖点点关注,收藏不迷路💖

width=“100%”>



💖The Start💖点点关注,收藏不迷路💖





相关文章:

React与原生事件:核心差异与性能对比解析

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐&#xff1a;「storms…...

Go 并发编程基础:select 多路复用

select 是 Go 并发编程中非常强大的语法结构&#xff0c;它允许程序同时等待多个通道操作的完成&#xff0c;从而实现多路复用机制&#xff0c;是协程调度、超时控制、通道竞争等场景的核心工具。 一、什么是 select select 类似于 switch 语句&#xff0c;但它用于监听多个通…...

暴雨新专利解决服务器噪音与性能悖论

6月1日&#xff0c;我国首部数据中心绿色化评价方面国家标准《绿色数据中心评价》正式实施&#xff0c;为我国数据中心的绿色低碳建设提供了明确指引。《评价》首次将噪音控制纳入国家级绿色评价体系&#xff0c;要求从设计隔声结构到运维定期监测实现闭环管控&#xff0c;加速…...

Go 语言中的内置运算符

1. 算术运算符 注意&#xff1a; &#xff08;自增&#xff09;和--&#xff08;自减&#xff09;在 Go 语言中是单独的语句&#xff0c;并不是运算符。 package mainimport "fmt"func main() {fmt.Println("103", 103) // 13fmt.Println("10-3…...

Spring Boot 中实现 HTTPS 加密通信及常见问题排查指南

Spring Boot 中实现 HTTPS 加密通信及常见问题排查指南 在金融行业安全审计中&#xff0c;未启用HTTPS的Web应用被列为高危漏洞。通过正确配置HTTPS&#xff0c;可将中间人攻击风险降低98%——本文将全面解析Spring Boot中HTTPS的实现方案与实战避坑指南。 一、HTTPS 核心原理与…...

项目研究:使用 LangGraph 构建智能客服代理

概述 本教程展示了如何使用 LangGraph 构建一个智能客服代理。LangGraph 是一个强大的工具&#xff0c;可用于构建复杂的语言模型工作流。该代理可以自动分类用户问题、分析情绪&#xff0c;并根据需要生成回应或升级处理。 背景动机 在当今节奏飞快的商业环境中&#xff0c…...

JS面试常见问题——数据类型篇

这几周在进行系统的复习&#xff0c;这一篇来说一下自己复习的JS数据结构的常见面试题中比较重要的一部分 文章目录 一、JavaScript有哪些数据类型二、数据类型检测的方法1. typeof2. instanceof3. constructor4. Object.prototype.toString.call()5. type null会被判断为Obje…...

创客匠人:如何通过创始人IP打造实现知识变现与IP变现的长效增长?

在流量红利逐渐消退的当下&#xff0c;创始人IP的价值愈发凸显。它不仅能够帮助中小企业及个人创业者突破竞争壁垒&#xff0c;还能成为企业品牌影响力的核心资产。然而&#xff0c;市场上IP孵化机构鱼龙混杂&#xff0c;如何选择一家真正具备长期价值的合作伙伴&#xff1f;创…...

【靶场】XXE-Lab xxe漏洞

前言 学习xxe漏洞,搭了个XXE-Lab的靶场 一、搭建靶场 现在需要登录,不知道用户名密码,先随便试试抓包 二、判断是否存在xxe漏洞 1.首先登录抓包 看到xml数据解析,由此判断和xxe漏洞有关,但还不确定xxe漏洞是否存在。 2.尝试xxe 漏洞 判断是否存在xxe漏洞 A.send to …...

开源项目实战学习之YOLO11:12.6 ultralytics-models-tiny_encoder.py

👉 欢迎关注,了解更多精彩内容 👉 欢迎关注,了解更多精彩内容 👉 欢迎关注,了解更多精彩内容 ultralytics-models-sam 1.sam-modules-tiny_encoder.py2.数据处理流程3.代码架构图(类层次与依赖)blocks.py: 定义模型中的各种模块结构 ,如卷积块、残差块等基础构建…...

Python[数据结构及算法 --- 栈]

一.栈的概念 在 Python 中&#xff0c;栈&#xff08;Stack&#xff09;是一种 “ 后进先出&#xff08;LIFO&#xff09;”的数据结构&#xff0c;仅允许在栈顶进行插入&#xff08;push&#xff09;和删除&#xff08;pop&#xff09;操作。 二.栈的抽象数据类型 1.抽象数…...

Unity VR/MR开发-开发环境准备

视频讲解链接&#xff1a; 【XR马斯维】UnityVR/MR开发环境准备【UnityVR/MR开发教程--入门】_哔哩哔哩_bilibili...

2025-06-08-深度学习网络介绍(语义分割,实例分割,目标检测)

深度学习网络介绍(语义分割,实例分割,目标检测) 前言 在开始这篇文章之前&#xff0c;我们得首先弄明白&#xff0c;什么是图像分割&#xff1f; 我们知道一个图像只不过是许多像素的集合。图像分割分类是对图像中属于特定类别的像素进行分类的过程&#xff0c;即像素级别的…...

Caliper 配置文件解析:config.yaml 和 fisco-bcos.json 附加在caliper中执行不同的合约方法

Caliper 配置文件解析:config.yaml 和 fisco-bcos.json Caliper 是一个区块链性能基准测试工具,用于评估不同区块链平台的性能。下面我将详细解释你提供的 fisco-bcos.json 文件结构,并说明它与 config.yaml 文件的关系。 fisco-bcos.json 文件解析 这个文件是针对 FISCO…...

【Ragflow】26.RagflowPlus(v0.4.0):完善解析逻辑/文档撰写模式全新升级

概述 在历经半个月的间歇性开发后&#xff0c;RagflowPlus再次迎来一轮升级&#xff0c;正式发布v0.4.0。 开源地址&#xff1a;https://github.com/zstar1003/ragflow-plus 更新方法 下载仓库最新代码&#xff1a; git clone https://github.com/zstar1003/ragflow-plus.…...

智能照明系统:具备认知能力的“光神经网络”

智能照明系统是物联网技术与传统照明深度融合的产物&#xff0c;其本质是通过感知环境、解析需求、自主决策的闭环控制&#xff0c;重构光与人、空间、环境的关系。这一系统由智能光源、多维传感器、边缘计算单元及云端管理平台构成&#xff0c;形成具备认知能力的“光神经网络…...

ubuntu系统 | docker+dify+ollama+deepseek搭建本地应用

1、docker 介绍与安装 docker安装:1、Ubuntu系统安装docker_ubuntu docker run-CSDN博客 docker介绍及镜像源配置:2、ubuntu系统docker介绍及镜像源和仓库配置-CSDN博客 docker常用命令:3、ubuntu系统docker常用命令-CSDN博客 docker compose安装:4、docker compose-CS…...

Docker 镜像上传到 AWS ECR:从构建到推送的全流程

一、在 EC2 实例中安装 Docker&#xff08;适用于 Amazon Linux 2&#xff09; 步骤 1&#xff1a;连接到 EC2 实例 ssh -i your-key.pem ec2-useryour-ec2-public-ip步骤 2&#xff1a;安装 Docker sudo yum update -y sudo amazon-linux-extras enable docker sudo yum in…...

SpringSecurity+vue通用权限系统

SpringSecurityvue通用权限系统 采用主流的技术栈实现&#xff0c;Mysql数据库&#xff0c;SpringBoot2Mybatis Plus后端&#xff0c;redis缓存&#xff0c;安全框架 SpringSecurity &#xff0c;Vue3.2Element Plus实现后台管理。基于JWT技术实现前后端分离。项目开发同时采 …...

如何在Spring Boot中使用注解动态切换实现

还在用冗长的if-else或switch语句管理多个服务实现? 相信不少Spring Boot开发者都遇到过这样的场景:需要根据不同条件动态选择不同的服务实现。 如果告诉你可以完全摆脱条件判断,让Spring自动选择合适的实现——只需要一个注解,你是否感兴趣? 本文将详细介绍这种优雅的…...

短视频时长预估算法调研

weighted LR o d d s T p 1 − p ( 1 − p ) o d d s T p ( T p o d d s ∗ p ) o d d s p o d d s T o d d s odds \frac{Tp}{1-p} \newline (1-p)odds Tp \newline (Tp odds * p) odds \newline p \frac{odds}{T odds} \newline odds1−pTp​(1−p)oddsTp(Tpodds…...

基于Java的离散数学题库系统设计与实现:附完整源码与论文

JAVASQL离散数学题库管理系统 一、系统概述 本系统采用Java Swing开发桌面应用&#xff0c;结合SQL Server数据库实现离散数学题库的高效管理。系统支持题型分类&#xff08;选择题、填空题、判断题等&#xff09;、难度分级、知识点关联&#xff0c;并提供智能组卷、在线测试…...

板凳-------Mysql cookbook学习 (十--2)

5.12 模式匹配中的大小写问题 mysql> use cookbook Database changed mysql> select a like A, a regexp A; ------------------------------ | a like A | a regexp A | ------------------------------ | 1 | 1 | --------------------------…...

设计模式域——软件设计模式全集

摘要 软件设计模式是软件工程领域中经过验证的、可复用的解决方案&#xff0c;旨在解决常见的软件设计问题。它们是软件开发经验的总结&#xff0c;能够帮助开发人员在设计阶段快速找到合适的解决方案&#xff0c;提高代码的可维护性、可扩展性和可复用性。设计模式主要分为三…...

FTPS、HTTPS、SMTPS以及WebSockets over TLS的概念及其应用场景

一、什么是FTPS&#xff1f; FTPS&#xff0c;英文全称File Transfer Protocol with support for Transport Layer Security (SSL/TLS)&#xff0c;安全文件传输协议&#xff0c;是一种对常用的文件传输协议(FTP)添加传输层安全(TLS)和安全套接层(SSL)加密协议支持的扩展协议。…...

RK3568项目(七)--uboot系统之外设与PMIC详解

目录 一、引言 二、按键 ------>2.1、按键种类 ------------>2.1.1、RESET ------------>2.1.2、UPDATE ------------>2.1.3、PWRON 部分 ------------>2.1.4、RK809 PMIC ------------>2.1.5、ADC按键 ------------>2.1.6、ADC按键驱动 ------…...

Three.js进阶之粒子系统(一)

一些特定模糊现象&#xff0c;经常使用粒子系统模拟&#xff0c;如火焰、爆炸等。Three.js提供了多种粒子系统&#xff0c;下面介绍粒子系统 一、Sprite粒子系统 使用场景&#xff1a;下雨、下雪、烟花 ce使用代码&#xff1a; var materialnew THRESS.SpriteMaterial();//…...

【仿生机器人】刀剑神域——爱丽丝苏醒计划,需求文档

仿生机器人"爱丽丝"系统架构设计需求文档 一、硬件基础 已完成头部和颈部硬件搭建 25个舵机驱动表情系统 颈部旋转功能 眼部摄像头&#xff08;视觉输入&#xff09; 麦克风阵列&#xff08;听觉输入&#xff09; 颈部发声装置&#xff08;语音输出&#xff09…...

小白的进阶之路系列之十四----人工智能从初步到精通pytorch综合运用的讲解第七部分

通过示例学习PyTorch 本教程通过独立的示例介绍PyTorch的基本概念。 PyTorch的核心提供了两个主要特性: 一个n维张量,类似于numpy,但可以在gpu上运行 用于构建和训练神经网络的自动微分 我们将使用一个三阶多项式来拟合问题 y = s i n ( x ) y=sin(x) y=sin(x),作为我们的…...

JavaScript性能优化实战大纲

性能优化的核心目标 降低页面加载时间&#xff0c;减少内存占用&#xff0c;提高代码执行效率&#xff0c;确保流畅的用户体验。 代码层面的优化 减少全局变量使用&#xff0c;避免内存泄漏 // 不好的实践 var globalVar I am global;// 好的实践 (function() {var localV…...