[vue3] patchFlags与位运算

Vue3在编译template的过程中会分析模板中的动态部分和静态部分,并标记相应的flag,用于在运行时优化虚拟DOM的更新。

  • Parse:将模板字符串解析成AST
  • Transform:对AST进行转换和优化,包括识别动态节点和静态节点;
  • CodeGeneration:将转换后的AST生成渲染函数,这个阶段会生成patchFlags

diff过程中,遇到包含dynamicChildren的块时,diff算法会进入优化模式,跳过对静态节点的处理从而优化了diff的执行效率。

flag的种类

源码位置:core/packages/shared/src/patchFlags.ts at main · vuejs/core (github.com)

export enum PatchFlags {TEXT = 1,CLASS = 1 << 1,STYLE = 1 << 2,PROPS = 1 << 3,FULL_PROPS = 1 << 4,NEED_HYDRATION = 1 << 5,STABLE_FRAGMENT = 1 << 6,KEYED_FRAGMENT = 1 << 7,UNKEYED_FRAGMENT = 1 << 8,NEED_PATCH = 1 << 9,DYNAMIC_SLOTS = 1 << 10,DEV_ROOT_FRAGMENT = 1 << 11,HOISTED = -1,BAIL = -2,
}

可以看到flag使用二进制格式记录的,并且每个标志仅有一位为1,这样可以通过位运算获知一个复合状态里包含哪些状态。

flag含义

  • TEXT:表示元素具有动态的 textContent

    <div>{{ dynamicText }}</div>
    
  • CLASS:表示元素具有动态的类绑定。

    <template><div :class="dynamicClass">Content</div>
    </template><script>
    export default {data() {return {dynamicClass: 'active'}}
    }
    </script>
    
  • STYLE:表示元素具有动态样式。

    <template><div :style="dynamicStyle">Content</div>
    </template><script>
    export default {data() {return {dynamicStyle: {color: 'red'}}}
    }
    </script>
    
  • PROPS:表示元素具有非 class/style 的动态属性。也可以用于具有任何动态属性的组件。

    <template><input :value="dynamicValue" />
    </template><script>
    export default {data() {return {dynamicValue: 'Hello'}}
    }
    </script>
    
  • FULL_PROPS:表示具有动态键属性的元素。当键变化时,总是需要完全差异检查。

    <template><div v-bind:[dynamicProp]="dynamicValue">Content</div>
    </template><script>
    export default {data() {return {dynamicProp: 'id',dynamicValue: 'uniqueId'}}
    }
    </script>
    
  • NEED_HYDRATION:表示该元素在客户端渲染时,需要将属性从静态 HTML转换为动态绑定。hydration是指从服务器端渲染(SSR)的静态内容中恢复出动态行为和状态的过程。该元素不需要常规的虚拟 DOM 属性更新,只需要在初始化时处理特定的属性以恢复其动态行为。

    案例(事件监听器):如 @click="handler",在服务器端渲染时,事件绑定不会被实际添加,客户端加载后需要将事件监听器正确绑定到元素上。

    <template><button @click="handleClick">Click me</button>
    </template><script>
    export default {methods: {handleClick() {console.log('Button clicked!')}}
    }
    </script>
    
  • STABLE_FRAGMENT:表示子元素顺序不变的片段。

  • KEYED_FRAGMENT:表示子元素的都带有或部分带有key标注。

  • UNKEYED_FRAGMENT:表示子元素没有key标注的片段。

  • NEED_PATCH:表示不涉及classstyleprops但仍需触发更新的情况,通常对应ref、指令等使用场景。

  • DYNAMIC_SLOTS:主要用于标识那些插槽内容或插槽名称是动态变化的组件。带有此标志的组件在更新时会被强制更新,以确保插槽内容或名称的变化能够正确反映到 DOM 中。

    <template><parent-component><template :slot="dynamicSlotName"><child-component :data="someData" /></template></parent-component>
    </template><script>
    export default {data() {return {dynamicSlotName: 'defaultSlot',someData: { message: 'Hello, World!' }}}
    }
    </script>
    
  • DEV_ROOT_FRAGMENT:表示用户在template的顶层写了注释而创建的flag。仅用于开发环境,因为生产中会去除注释。

    <template><!-- Root level comment --><div>Content</div>
    </template>
    
  • HOISTED:表示提升的静态虚拟节点。patch过程可以跳过整个子树,因为静态内容永远不需要更新。

    <p>Static content</p>
    
  • BAIL:表示diff算法应退出优化模式,通常是对应用户使用h函数自定义渲染函数的情况。

示例代码

vue3有提供一个playground可以查看编译后的结果:Vue SFC Playground (vuejs.org)

简单的代码案例

<script setup>
import { ref } from 'vue'const msg = ref('Hello World!')
</script><template><p>static content</p><h1>{{ msg }}</h1><input v-model="msg" />
</template>

编译后的JS

import { ref } from 'vue'const __sfc__ = {__name: 'App',setup(__props, { expose: __expose }) {__expose();const msg = ref('Hello World!')const __returned__ = { msg, ref }
Object.defineProperty(__returned__, '__isScriptSetup', { enumerable: false, value: true })
return __returned__
}};
import { createElementVNode as _createElementVNode, toDisplayString as _toDisplayString, vModelText as _vModelText, withDirectives as _withDirectives, Fragment as _Fragment, openBlock as _openBlock, createElementBlock as _createElementBlock } from "vue"const _hoisted_1 = /*#__PURE__*/_createElementVNode("p", null, "static content", -1 /* HOISTED */)
function render(_ctx, _cache, $props, $setup, $data, $options) {return (_openBlock(), _createElementBlock(_Fragment, null, [_hoisted_1,_createElementVNode("h1", null, _toDisplayString($setup.msg), 1 /* TEXT */),_withDirectives(_createElementVNode("input", {"onUpdate:modelValue": _cache[0] || (_cache[0] = $event => (($setup.msg) = $event))}, null, 512 /* NEED_PATCH */), [[_vModelText, $setup.msg]])], 64 /* STABLE_FRAGMENT */))
}
__sfc__.render = render
__sfc__.__file = "src/App.vue"
export default __sfc__
  • 可以看到<p>static content</p>被应用了静态提升(Vue3优化策略之一):

(静态的flag-1

const _hoisted_1 = /*#__PURE__*/_createElementVNode("p", null, "static content", -1 /* HOISTED */)
  • <h1>{{ msg }}</h1>由于有动态文本,被标记为TEXT
  • <input v-model="msg" />使用了v-model指令,被标记了NEED_PATCH
  • Vue2template内部只能存在一个顶级节点,如果有多个要使用一个标签囊括其中;Vue3支持template内部多个顶级节点,其实是框架帮我们套了一个fragment;在上述代码中由于这个fragment内部元素的顺序是固定的,因此被标记为STABLE_FRAGMENT

静态提升

静态提升是Vue3的一种性能优化手段。如果有VNode被标记为静态节点,说明它的内容是固定不变的。那么它的构建函数会被提升到渲染函数的外部,即只会被运行一次。

位运算的应用

Vue3中,这些flags都是只有一位为1,在这个前提下,可以通过位运算实现下面两种操作:

组合标志

通过或运算组合标志:

const combinedFlag = PatchFlags.TEXT | PatchFlags.STYLE; // 0001 | 0100 = 0101

检查标志

通过与运算检查混合标志是否存在某个base flag

const hasText = combinedFlag & PatchFlags.TEXT; // 0101 & 0001 = 0001 (truthy)
const hasClass = combinedFlag & PatchFlags.CLASS; // 0101 & 0010 = 0000 (falsy)

可以在源码中看到patchFlag和与运算的相关代码:core/packages/runtime-core/src/renderer.ts at main · vuejs/core (github.com)

image-20240803181837861

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/777290.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

进度报告10

(1)1.代码块2.继承:用extends关键字,不支持多继承

2024-08-03:用go语言,给定一个从 0 开始的字符串数组 `words`, 我们定义一个名为 `isPrefixAndSuffix` 的布尔函数,该函数接受两个字符串参数 `str1` 和

2024-08-03:用go语言,给定一个从 0 开始的字符串数组 words, 我们定义一个名为 isPrefixAndSuffix 的布尔函数,该函数接受两个字符串参数 str1 和 str2。 当 str1 同时是 str2 的前缀和后缀时,函数返回 true;否则返回 false。 例如,isPrefixAndSuffix("aba", …

AGC064B 题解

设红色的点值为 0,蓝色为 1。 注意到,如果有一条边的颜色和两端点同色,一定可以选。例子:选择和两端点同色的边。 又发现,如果存在一个 \(sz>1\) 的合法连通块,无论和其他点怎么连,原来的这个连通块内的点一定合法。 有注意到形如 \(0\xleftrightarrow 10,1\xleftrig…

Java源码:泡泡堂

学习java朋友们,福利来了,今天小编给大家带来了一款泡泡堂源码。此源码仅供学习使用!! 源码搭建和讲解 代码简介包名称类描述com.whh.frame MyJFrame.java 显示窗体、绑定监听、启动线程com.whh.frame MyJPanel.java 不断读取人物信息com.whh.main GameStart.java 游戏入口…

24暑假集训day5上午

图论 差分约束 有 \(𝑛\) 个整数变量 \(𝑥_1∼𝑥_𝑛\)。 给定一些形如 \(𝑥_𝑖+𝑐≥𝑥_𝑗\) 的限制。问有没有可行解,如有输出方案。 例如 \(𝑥_1−1≥𝑥_2,𝑥_2≥𝑥_3,𝑥_3≥𝑥_1\) 就无解。 在单源最短路问题中,如果存在一条 \(𝑖→𝑗\)…

24暑假集训day4上午下午

基础图论 图的存储方式 无向边可以拆成两条有向边 1. 邻接矩阵 邻接矩阵:若 \(𝑖→𝑗\) 存在有向边,则令矩阵 \(𝐴[𝑖][𝑗]=1\)。 遍历一个点的所有出边是 \(𝑂(𝑛)\) 的。 空间复杂度 \(𝑂(𝑛^2 )\)。 总结:复杂度太高,尽量不使用 bool hasEdge[MAXN][M…

微客在线客服系统-连接一切客户

嘿,朋友们,我是微客客服,一个全能的在线客服小能手。今天,我要给大家秀一秀我的超能力——多渠道对接! 想象一下,你的企业就像一个繁忙的交通枢纽,客户咨询就像来来往往的车辆,而我,就是那个指挥交通的智能信号灯。我能帮你把客户咨询引导到正确的地方,让一切沟通都井…

Overleaf中插入pdf图片只显示图片路径的解决方式

最近在用Overleaf写一篇论文,使用IEEE的LaTex模板时发现一个问题,我使用pdfLaTex编译器无法正确显示我插入的pdf图片,网上翻解决方式没有翻到,误打误撞解决了这个问题,问题如下图所示: 即只在图片区域显示路径,不显示图片本身,解决方案是: 在右侧设置里找到编译模式,…

RHCSA 考试试题解析

1.登录测试机初始化 red bule主机 rht-vmctl reset red rht-vmctl reset blue

MySQL的执行计划详解(Explain)(整合版)

目录MySQL的执行计划详解(Explain)(整合版)1、MySQL执行计划的定义2、Explain分析示例3、语法展示4、explain中的列4.1、id4.2、select_type4.3、table4.4、type(重要的)4.5、possible_keys4.6、key4.7、key_len4.8、ref4.9、rows4.10、fitered4.11、Extra MySQL的执行计划详…

ArkTS #01# 组件通信

一、通过Prop单向传递/* * 单双向绑定都有 * 父组件@State,子组件@Link,孙组件@Prop * 数据流向:父组件 <--> 子组件 --> 孙组件*/@Entry @Component export struct BothBinding {@State fatherValue: number = 0//表示组件中的状态变量,这个状态变化会引起 UI 变…

风风影视安装使用

风风影视安装使用教程 一 安装软件下载风风影视 下载好后安装即可。二 配置地址1.点击设置 2.点击配置地址 3.将这串地址复制粘贴,点击确认https://gitee.com/ffddoz/tvbox1/raw/master/fantaiyingff.json到这步就可以使用了以下是功能介绍,非必须看点击主页之后,会有很多选…