v-on的思考

某位面试官为了一个问题, v-on可以绑定几个函数

正常来想, v-on不是只一个函数吗, 还可以绑定多个?

正确答案: 无数个

vue 复制代码
<div @click="handle1(), handle2(), handle3() ..."></div>

<!-- or -->

<div @click="[handle1(), handle2(), handle3()]"></div>

<!-- but 上面的方法中不写括号是不行的 -->

为什么会成这样的问题呢?

看文档

文档上说

md 复制代码
v-on​

给元素绑定事件监听器。

缩写:@

期望的绑定值类型:`Function | Inline Statement | Object (不带参数)`

类型中的 Function可以理解, 就是最常用的直接传递函数

Object类型主要指的是一个v-on绑定多个事件(不常用)

vue 复制代码
<!-- 对象语法 -->
<button v-on="{ mousedown: doThis, mouseup: doThat }"></button>

那么这个Inline Statement是个什么类型, 这也是这里主要探讨的地方

看编译

详情

vue 复制代码
<div @click="handle1(), handle2(), handle3() ..."></div>
js 复制代码
// 完整编译结果, 后面省略重复部分 // [!code focus]
_createElementVNode("button", {
  onClick: _cache[0] || (_cache[0] = $event => ($setup.handle1(), $setup.handle2(), $setup.handle3())) // [!code focus]
}),

其实 <div @click="handle1(), handle2(), handle3() ..."></div> 就是上面的 Inline Statement 类型

编译器会将 handle1(), handle2(), handle3() ... 当作一个语句, 然后再触发事件是运行这个语句

数组写法同理

vue 复制代码
<div @click="[handle1(), handle2(), handle3()]"></div>
js 复制代码
// 数组编译结果
;($event) => [$setup.handle1(), $setup.handle2(), $setup.handle3()]

甚至可以在里面写任何东西

vue 复制代码
<div @click="{ handle1 }"></div>
js 复制代码
// 编译结果
;($event) => ({ handle1: $setup.handle1 })

这也解释了为什么不加括号不生效的原因, 因为当这个事件触发时, 语句运行了, 但是函数因为没有括号而没有被调用

js 复制代码
// 无括号编译结果
;($event) => ($setup.handle1, $setup.handle2, $setup.handle3)

为什么编译后的结果是这样的呢

看源码

差不多的意思就是说, 如果判断出来是一个inline statement, 那么就拼接一个函数, inline statement就是函数体

顺便一提, inline statement 支持多行, 意思就是说, 这里可以写分号, 编译出来就是多行

vue 复制代码
<!-- prettier-ignore -->
<div @click="handle1(); handle2(); handle3()"></div>
js 复制代码
// 编译结果
// prettier-ignore
;$event => {$setup.log(); $setup.log()}

思考

所以为什么如果只传函数的话可以加括号也可以不加

因为如果加了括号就是一个inline statement

js 复制代码
;($event) => $setup.handle1()

如果不加括号就是一个函数, 会把这个函数直接传递个事件

js 复制代码
_createElementVNode('button', { onClick: $setup.log })
相关推荐
铭毅天下几秒前
EasySearch Rules 规则语法速查手册
开发语言·前端·javascript·ecmascript
GISer_Jing12 分钟前
AI Agent操作系统架构师:Harness Engineer解析
前端·人工智能·ai·aigc
英俊潇洒美少年21 分钟前
css中专门用来提升渲染性能、减少重排重绘的属性
前端·css
天若有情67334 分钟前
前端HTML精讲01:别再乱 div 一把抓,吃透语义化标签才是进阶第一步
前端·html
Highcharts.js35 分钟前
React 开发者的图表库生态:Highcharts React
前端·react.js·前端框架
阿部多瑞 ABU35 分钟前
文明文化悖论
前端·人工智能·ai写作
钛态1 小时前
Flutter 三方库 react 泛前端核心范式框架鸿蒙原生层生态级双向超能适配:跨时空重塑响应式单向数据流拓扑与高度精密生命周期树引擎解耦视图渲染控制中枢(适配鸿蒙 HarmonyOS ohos)
前端·flutter·react.js
全栈前端老曹1 小时前
【前端地图】地图开发基础概念——地图服务类型(矢量图、卫星图、地形图)、WGS84 / GCJ-02 / BD09 坐标系、地图 SDK 简介
前端·javascript·地图·wgs84·gcj-02·bd09·地图sdk
只与明月听1 小时前
RAG深入学习之向量数据库
前端·人工智能·python
吕不说1 小时前
AI 面试总挂?可能是表达出了问题:三层表达法 + STAR 进阶框架
前端