找回密码
 立即注册
首页 业界区 业界 Vue3 中的 v-bind 指令:你不知道的那些工作原理 ...

Vue3 中的 v-bind 指令:你不知道的那些工作原理

神泱 2025-6-6 15:23:59
前言

v-bind指令想必大家都不陌生,并且都知道他支持各种写法,比如、、(vue3.4中引入的新的写法)。这三种写法的作用都是一样的,将title变量绑定到div标签的title属性上。本文将通过debug源码的方式带你搞清楚,v-bind指令是如何实现这么多种方式将title变量绑定到div标签的title属性上的。注:本文中使用的vue版本为3.4.19。
关注公众号:【前端欧阳】,给自己一个进阶vue的机会
看个demo

还是老套路,我们来写个demo。代码如下:
  1. <template>
  2.   Hello Word
  3.   Hello Word
  4.   Hello Word
  5. </template>
复制代码
上面的代码很简单,使用三种写法将title变量绑定到div标签的title属性上。
我们从浏览器中来看看编译后的代码,如下:
  1. const _sfc_main = _defineComponent({
  2.   __name: "index",
  3.   setup(__props, { expose: __expose }) {
  4.     // ...省略
  5.   }
  6. });
  7. function _sfc_render(_ctx, _cache, $props, $setup, $data, $options) {
  8.   return _openBlock(), _createElementBlock(
  9.     _Fragment,
  10.     null,
  11.     [
  12.       _createElementVNode("div", { title: $setup.title }, "Hello Word", 8, _hoisted_1),
  13.       _createElementVNode("div", { title: $setup.title }, "Hello Word", 8, _hoisted_2),
  14.       _createElementVNode("div", { title: $setup.title }, "Hello Word", 8, _hoisted_3)
  15.     ],
  16.     64
  17.     /* STABLE_FRAGMENT */
  18.   );
  19. }
  20. _sfc_main.render = _sfc_render;
  21. export default _sfc_main;
复制代码
从上面的render函数中可以看到三种写法生成的props对象都是一样的: { title: $setup.title }。props属性的key为title,值为$setup.title变量。
再来看看浏览器渲染后的样子,如下图:
1.png

从上图中可以看到三个div标签上面都有title属性,并且属性值都是一样的。
transformElement函数

在之前的 面试官:来说说vue3是怎么处理内置的v-for、v-model等指令?文章中我们讲过了在编译阶段会执行一堆transform转换函数,用于处理vue内置的v-for等指令。而v-bind指令就是在这一堆transform转换函数中的transformElement函数中处理的。
还是一样的套路启动一个debug终端。这里以vscode举例,打开终端然后点击终端中的+号旁边的下拉箭头,在下拉中点击Javascript Debug Terminal就可以启动一个debug终端。
2.png

给transformElement函数打个断点,transformElement函数的代码位置在:node_modules/@vue/compiler-core/dist/compiler-core.cjs.js。
在debug终端上面执行yarn dev后在浏览器中打开对应的页面,比如:http://localhost:5173/ 。此时断点就会走到transformElement函数中,在我们这个场景中简化后的transformElement函数代码如下:
  1. const transformElement = (node, context) => {
  2.   return function postTransformElement() {
  3.     let vnodeProps;
  4.     const propsBuildResult = buildProps(
  5.       node,
  6.       context,
  7.       undefined,
  8.       isComponent,
  9.       isDynamicComponent
  10.     );
  11.     vnodeProps = propsBuildResult.props;
  12.     node.codegenNode = createVNodeCall(
  13.       context,
  14.       vnodeTag,
  15.       vnodeProps,
  16.       vnodeChildren
  17.       // ...省略
  18.     );
  19.   };
  20. };
复制代码
我们先来看看第一个参数node,如下图:
3.png

从上图中可以看到此时的node节点对应的就是Hello Word节点,其中的props数组中只有一项,对应的就是div标签中的v-bind:title="title"部分。
我们接着来看transformElement函数中的代码,可以分为两部分。
第一部分为调用buildProps函数拿到当前node节点的props属性赋值给vnodeProps变量。
第二部分为根据当前node节点vnodeTag也就是节点的标签比如div、vnodeProps也就是节点的props属性对象、vnodeChildren也就是节点的children子节点、还有一些其他信息生成codegenNode属性。在之前的 终于搞懂了!原来 Vue 3 的 generate 是这样生成 render 函数的文章中我们已经讲过了编译阶段最终生成render函数就是读取每个node节点的codegenNode属性然后进行字符串拼接。
从buildProps函数的名字我们不难猜出他的作用就是生成node节点的props属性对象,所以我们接下来需要将目光聚焦到buildProps函数中,看看是如何生成props对象的。
buildProps函数

将断点走进buildProps函数,在我们这个场景中简化后的代码如下:
  1. function buildProps(node, context, props = node.props) {
  2.   let propsExpression;
  3.   let properties = [];
  4.   for (let i = 0; i < props.length; i++) {
  5.     const prop = props[i];
  6.     const { name } = prop;
  7.     const directiveTransform = context.directiveTransforms[name];
  8.     if (directiveTransform) {
  9.       const { props } = directiveTransform(prop, node, context);
  10.       properties.push(...props);
  11.     }
  12.   }
  13.   propsExpression = createObjectExpression(
  14.     dedupeProperties(properties),
  15.     elementLoc
  16.   );
  17.   return {
  18.     props: propsExpression,
  19.     // ...省略
  20.   };
  21. }
复制代码
由于我们在调用buildProps函数时传的第三个参数为undefined,所以这里的props就是默认值node.props。如下图:
4.png

从上图中可以看到props数组中只有一项,props中的name字段为bind,说明v-bind指令还未被处理掉。
并且由于我们当前node节点是第一个div标签:,所以props中的rawName的值是v-bind:title。
我们接着来看上面for循环遍历props的代码:const directiveTransform = context.directiveTransforms[name],现在我们已经知道了这里的name为bind。那么这里的context.directiveTransforms对象又是什么东西呢?我们在debug终端来看看context.directiveTransforms,如下图:
5.png

从上图中可以看到context.directiveTransforms对象中包含许多指令的转换函数,比如v-bind、v-cloak、v-html、v-model等。
我们这里name的值为bind,并且context.directiveTransforms对象中有name为bind的转换函数。所以const directiveTransform = context.directiveTransforms[name]就是拿到处理v-bind指令的转换函数,然后赋值给本地的directiveTransform函数。
接着就是执行directiveTransform转换函数,拿到v-bind指令生成的props数组。然后执行properties.push(...props)方法将所有的props数组都收集到properties数组中。
由于node节点中有多个props,在for循环遍历props数组时,会将经过transform转换函数处理后拿到的props数组全部push到properties数组中。properties数组中可能会有重复的prop,所以需要执行dedupeProperties(properties)函数对props属性进行去重。
node节点上的props属性本身也是一种node节点,所以最后就是执行createObjectExpression函数生成props属性的node节点,代码如下:
  1. propsExpression = createObjectExpression(
  2.   dedupeProperties(properties),
  3.   elementLoc
  4. )
复制代码
其中createObjectExpression函数的代码也很简单,代码如下:
  1. function createObjectExpression(properties, loc) {
  2.   return {
  3.     type: NodeTypes.JS_OBJECT_EXPRESSION,
  4.     loc,
  5.     properties,
  6.   };
  7. }
复制代码
上面的代码很简单,properties数组就是node节点上的props数组,根据properties数组生成props属性对应的node节点。
我们在debug终端来看看最终生成的props对象propsExpression是什么样的,如下图:
6.png

从上图中可以看到此时properties属性数组中已经没有了v-bind指令了,取而代之的是key和value属性。key.content的值为title,说明属性名为title。value.content的值为$setup.title,说明属性值为变量$setup.title。
到这里v-bind指令已经被完全解析了,生成的props对象中有key和value字段,分别代表的是属性名和属性值。后续生成render函数时只需要遍历所有的props,根据key和value字段进行字符串拼接就可以给div标签生成title属性了。
接下来我们继续来看看处理v-bind指令的transform转换函数具体是如何处理的。
transformBind函数

将断点走进transformBind函数,在我们这个场景中简化后的代码如下:
  1. const transformBind = (dir, _node) => {
  2.   const arg = dir.arg;
  3.   let { exp } = dir;
  4.   if (!exp) {
  5.     const propName = camelize(arg.content);
  6.     exp = dir.exp = createSimpleExpression(propName, false, arg.loc);
  7.     exp = dir.exp = processExpression(exp, context);
  8.   }
  9.   return {
  10.     props: [createObjectProperty(arg, exp)],
  11.   };
  12. };
复制代码
我们先来看看transformBind函数接收的第一个参数dir,从这个名字我想你应该已经猜到了他里面存储的是指令相关的信息。
在debug终端来看看三种写法的dir参数有什么不同。
第一种写法:的dir如下图:
7.png

从上图中可以看到dir.name的值为bind,说明这个是v-bind指令。dir.rawName的值为v-bind:title说明没有使用缩写模式。dir.arg表示bind绑定的属性名称,这里绑定的是title属性。dir.exp表示bind绑定的属性值,这里绑定的是$setup.title变量。
第二种写法:的dir如下图:
8.png

从上图中可以看到第二种写法的dir和第一种写法的dir只有一项不一样,那就是dir.rawName。在第二种写法中dir.rawName的值为:title,说明我们这里是采用了缩写模式。
可能有的小伙伴有疑问了,这里的dir是怎么来的?vue是怎么区分第一种全写模式和第二种缩写模式呢?
答案是在parse阶段将html编译成AST抽象语法树阶段时遇到v-bind:title和:title时都会将其当做v-bind指令处理,并且将解析处理的指令绑定的属性名塞到dir.arg中,将属性值塞到dir.exp中。
第三种写法:的dir如下图:
9.png

第三种写法也是缩写模式,并且将属性值也一起给省略了。所以这里的dir.exp存储的属性值为undefined。其他的和第二种缩写模式基本一样。
我们再来看transformBind中的代码,if (!exp)说明将值也一起省略了,是第三种写法。就会执行如下代码:
  1. if (!exp) {
  2.   const propName = camelize(arg.content);
  3.   exp = dir.exp = createSimpleExpression(propName, false, arg.loc);
  4.   exp = dir.exp = processExpression(exp, context);
  5. }
复制代码
这里的arg.content就是属性名title,执行camelize函数将其从kebab-case命名法转换为驼峰命名法。比如我们给div上面绑一个自定义属性data-type,采用第三种缩写模式就是这样的:。大家都知道变量名称是不能带短横线的,所以这里的要执行camelize函数将其转换为驼峰命名法:改为绑定dataType变量。
从前面的那几张dir变量的图我们知道 dir.exp变量的值是一个对象,所以这里需要执行createSimpleExpression函数将省略的变量值也补全。createSimpleExpression的函数代码如下:
  1. function createSimpleExpression(
  2.   content,
  3.   isStatic,
  4.   loc,
  5.   constType
  6. ): SimpleExpressionNode {
  7.   return {
  8.     type: NodeTypes.SIMPLE_EXPRESSION,
  9.     loc,
  10.     content,
  11.     isStatic,
  12.     constType: isStatic ? ConstantTypes.CAN_STRINGIFY : constType,
  13.   };
  14. }
复制代码
经过这一步处理后 dir.exp变量的值如下图:
10.png

还记得前面两种模式的 dir.exp.content的值吗?他的值是$setup.title,表示属性值为setup中定义的title变量。而我们这里的dir.exp.content的值为title变量,很明显是不对的。
所以需要执行exp = dir.exp = processExpression(exp, context)将dir.exp.content中的值替换为$setup.title,执行processExpression函数后的dir.exp变量的值如下图:
11.png

我们来看transformBind函数中的最后一块return的代码:
  1. return {
  2.   props: [createObjectProperty(arg, exp)],
  3. }
复制代码
这里的arg就是v-bind绑定的属性名,exp就是v-bind绑定的属性值。createObjectProperty函数代码如下:
  1. function createObjectProperty(key, value) {
  2.   return {
  3.     type: NodeTypes.JS_PROPERTY,
  4.     loc: locStub,
  5.     key: isString(key) ? createSimpleExpression(key, true) : key,
  6.     value,
  7.   };
  8. }
复制代码
经过createObjectProperty函数的处理就会生成包含key、value属性的对象。key中存的是绑定的属性名,value中存的是绑定的属性值。
其实transformBind函数中做的事情很简单,解析出v-bind指令绑定的属性名称和属性值。如果发现v-bind指令没有绑定值,那么就说明当前v-bind将值也给省略掉了,绑定的属性和属性值同名才能这样写。然后根据属性名和属性值生成一个包含key、value键的props对象。后续生成render函数时只需要遍历所有的props,根据key和value字段进行字符串拼接就可以给div标签生成title属性了。
总结

在transform阶段处理vue内置的v-for、v-model等指令时会去执行一堆transform转换函数,其中有个transformElement转换函数中会去执行buildProps函数。
buildProps函数会去遍历当前node节点的所有props数组,此时的props中还是存的是v-bind指令,每个prop中存的是v-bind指令绑定的属性名和属性值。
在for循环遍历node节点的所有props时,每次都会执行transformBind转换函数。如果我们在写v-bind时将值也给省略了,此时v-bind指令绑定的属性值就是undefined。这时就需要将省略的属性值补回来,补回来的属性值的变量名称和属性名是一样的。
在transformBind转换函数的最后会根据属性名和属性值生成一个包含key、value键的props对象。key对应的就是属性名,value对应的就是属性值。后续生成render函数时只需要遍历所有的props,根据key和value字段进行字符串拼接就可以给div标签生成title属性了。
关注公众号:【前端欧阳】,给自己一个进阶vue的机会
12.jpeg


来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

相关推荐

您需要登录后才可以回帖 登录 | 立即注册