React 事件机制原理

React事件机制原理

与原生 DOM 事件相比,React 事件机制有着独特的设计理念和实现方式,理解这一机制对于开发者写出高效、可维护的 React 代码至关重要。

一、React 事件机制的本质

React 事件机制并非直接映射原生 DOM 事件,而是在原生事件的基础上构建了一套合成事件系统(SyntheticEvent)

以如下react事件为例

jsx 复制代码
export default function Test(){
    function handleClick(e){
        console.log(e.target)
	}
    return (
        <button onClick={handleClick}>点我</button>
    )
}

对于 <button onClick={handleClick}>点我</button>事件,React并不会将这个函数直接绑定到button上,而是:把这个事件处理函数注册给 React 内部系统,并记录下来:"这个按钮点击时应该调用哪个函数"。然后,React 在最顶层(比如 document)监听所有点击事件。

当用户点击了某个按钮时,浏览器原生事件对象冒泡到顶层。React拿到这个事件后,根据(event.target)找到对应的组件,然后找到组件中绑定onClick函数,最后调用这个函数,并传入对原生事件封装的 SyntheticEvent 对象。

二、什么是SyntheticEvent

  • SyntheticEvent

    在 React 中,当给一个组件(比如按钮 <button>)添加事件处理函数时,React 并没有直接使用浏览器原生的事件对象(如 MouseEvent, KeyboardEvent 等),而是提供了一个统一的包装器 ------ SyntheticEvent。它拥有与原生事件相同的接口,但屏蔽了不同浏览器之间的差异,确保你的代码可以在所有浏览器中表现一致。

    jsx 复制代码
    function MyButton() {
      function handleClick(event) {
        // event 是 SyntheticEvent 类型
        console.log('按钮被点击了!');
        console.log(event); // 可以打印出事件对象
      }
    
      return <button onClick={handleClick}>点我</button>;
    }

    通过SyntheticEvent所有事件处理方式统一,不管是点击、键盘输入还是表单变化,都可以用相同的方式处理。

  • SyntheticEvent 常见属性和方法

    属性/方法 说明
    event.target 触发事件的目标元素(比如点击的是哪个 input)
    event.type 事件类型(如 'click', 'change'
    event.preventDefault() 阻止默认行为(比如阻止表单提交)
    event.stopPropagation() 阻止事件冒泡
    event.nativeEvent 获取原始的浏览器事件对象(不推荐常用)

三、异步事件访问问题

由于SyntheticEvent 是池化的(event pool)。即React 不会为每次事件都创建一个新的事件对象,而是从一个"池子"里取出一个已有的对象来复用 。处理完事件后,这个对象会被回收回池子里,供下一次使用。这样做的目的是为了 减少内存分配和垃圾回收的压力,提高性能

  • 这意味着React 在当前事件处理完成后,就会清空这个对象的内容(回收它) ,当你在 setTimeout 这样的异步操作中访问 event.target 时,这个事件对象可能已经被清空了,在异步操作中不能保留事件对象

    jsx 复制代码
    function App() {
      const handleClick = (e) => {
        console.log('绑定执行',e.target);
        setTimeout(() => {
          console.log('异步函数执行',e.target);
        }, 2000)
      }
      return (
        <>
          <button onClick={handleClick}>click</button>
        </>
      )
    }

    结果如下:

  • 若要在异步中使用事件对象,可以用变量来存储数据

    jsx 复制代码
    function App() {
      const handleClick = (e) => {
        let temp = e
        console.log('绑定执行',e.target);
        setTimeout(() => {
          console.log('异步函数执行',temp.target);
        }, 2000)
      }
      return (
        <>
          <button onClick={handleClick}>click</button>
        </>
      )
    }

    效果如下:

相关推荐
2501_9209317014 分钟前
React Native鸿蒙跨平台实现推箱子游戏,完成玩家移动与箱子推动,当所有箱子都被推到目标位置时,玩家获胜
javascript·react native·react.js·游戏·ecmascript·harmonyos
layman052844 分钟前
webpack5 css-loader:从基础到原理
前端·css·webpack
半桔1 小时前
【前端小站】CSS 样式美学:从基础语法到界面精筑的实战宝典
前端·css·html
AI老李1 小时前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
_OP_CHEN1 小时前
【前端开发之CSS】(一)初识 CSS:网页化妆术的终极指南,新手也能轻松拿捏页面美化!
前端·css·html·网页开发·样式表·界面美化
啊哈一半醒1 小时前
CSS 主流布局
前端·css·css布局·标准流 浮动 定位·flex grid 响应式布局
PHP武器库1 小时前
ULUI:不止于按钮和菜单,一个专注于“业务组件”的纯 CSS 框架
前端·css
电商API_180079052471 小时前
第三方淘宝商品详情 API 全维度调用指南:从技术对接到生产落地
java·大数据·前端·数据库·人工智能·网络爬虫
晓晓莺歌1 小时前
vue3某一个路由切换,导致所有路由页面均变成空白页
前端·vue.js
Up九五小庞2 小时前
开源埋点分析平台 ClkLog 本地部署 + Web JS 埋点测试实战--九五小庞
前端·javascript·开源