Vue 事件系统

在模板里写一行 @click="handler",背后却是一套从编译器到渲染器、再到 DOM 事件再绑定的精密流水线。本文带你拆解事件在 Vue 内部的完整生命周期。

一、事件的「身份」:props 的一种

模板:

vue 复制代码
<p @click="clickHandler">text</p>

经过编译器后,事件被翻译成虚拟节点里的一个普通属性:

js 复制代码
const vnode = {
  type: 'p',
  props: {
    onClick: clickHandler   // 注意首字母大写
  },
  children: 'text'
}

因此,在渲染器眼里,事件与 idclass 没有本质区别,只是 key 以 on 开头。

二、首次挂载

渲染器在 patchProps 阶段发现 props key 以 on 开头,就进入事件分支:

js 复制代码
if (/^on/.test(key)) {
  const eventName = key.slice(2).toLowerCase() // onClick -> click
  el.addEventListener(eventName, nextValue)
}

此时事件处理器被直接绑定到 DOM 节点,首屏即生效。

三、更新策略

如果每次更新都「先 remove 再 add」,会带来无谓的性能损耗。Vue 采用 invoker 代理 解决这一问题:

  1. 在元素上新建一个隐藏属性 el._vei(Vue Event Invoker),它是一个对象。
  2. 对象的键是事件名,值是一个固定引用的代理函数 invoker
  3. invoker 内部持有真正的事件数组或函数,更新时只需替换引用,无需重新绑定。

核心逻辑:

js 复制代码
const invokers = el._vei || (el._vei = {})
let invoker = invokers[eventName]

if (!invoker) {
  invoker = invokers[eventName] = (e) => {
    const fns = invoker.value
    Array.isArray(fns) ? fns.forEach(fn => fn(e)) : fns(e)
  }
  el.addEventListener(eventName, invoker)
}
invoker.value = nextValue   // 指向最新处理器

nextValue 是空数组或 null 时,直接 removeEventListener 即可卸载。

四、多处理器与同类型多事件

模板支持数组语法:

vue 复制代码
<button @click="[fn1, fn2]" @contextmenu="fn3" />

invoker 内部检测到 Array.isArray(value),就循环执行所有函数,实现「同类型多处理器」零成本扩展。

总结

Vue 的事件系统用一句话概括:

事件即 props,props 即数据,数据变化即更新,更新即最小化 DOM 操作。

理解 invoker 代理模式,你就拥有了在源码层面定位事件问题、优化事件性能的全部钥匙。

相关推荐
永远的个初学者20 小时前
图片优化 上传图片压缩 npm包支持vue(react)框架开源插件 支持在线与本地
前端·vue.js·react.js
爱吃土豆的马铃薯ㅤㅤㅤㅤㅤㅤㅤㅤㅤ20 小时前
npm i / npm install 卡死不动解决方法
前端·npm·node.js
Kratzdisteln20 小时前
【Cursor _RubicsCube Diary 1】Node.js;npm;Vite
前端·npm·node.js
杰克尼20 小时前
vue_day04
前端·javascript·vue.js
明远湖之鱼21 小时前
浅入理解跨端渲染:从零实现 React DSL 跨端渲染机制
前端·react native·react.js
悟忧21 小时前
规避ProseMirror React渲染差异带来的BUG
前端
小皮虾1 天前
小程序云开发有类似 uniCloud 云对象的方案吗?有的兄弟,有的!
前端·javascript·小程序·云开发
QuantumLeap丶1 天前
《uni-app跨平台开发完全指南》- 05 - 基础组件使用
vue.js·微信小程序·uni-app
Android疑难杂症1 天前
鸿蒙Notification Kit通知服务开发快速指南
android·前端·harmonyos
T___T1 天前
全方位解释 JavaScript 执行机制(从底层到实战)
前端·面试