vue3 使用h进行动态渲染会损失性能吗

起因

实现动态搭建form表单,需要从dsl中组建元素渲染,我们要知道究竟动态函数究竟在patch时候差异多少

js 复制代码
// 最小demo
<component :is="renderFunc"></component>
const renderFunc = () => {
  return h('div', {innerHTML: props.a})
}

源码分析

  1. 首先我们先建立两个最小demo,通过perfomance和debug断点手段去分析
js 复制代码
app.vue

<template>
  <div>
    <button @click="handleChange"></button>
    <component :is="HelloWorld" :a="a"></component>
  </div>
</template>

<script setup>
import { ref } from 'vue';
import HelloWorld from './components/HelloWorld.vue'
const a = ref('1')
const handleChange = () => {
  a.value = Math.random().toString()
}

情况1, h函数

js 复制代码
HelloWorld.vue

<template>
 <component :is="renderFunc"></component>
</template>

<script setup>
import { h } from 'vue';
/* eslint-disable */
const props = defineProps({
  a: String
})

const renderFunc = () => {
  return h('div', {innerHTML: props.a})
}
</script>

情况2, 组件

js 复制代码
HelloWorld.vue

<template>
<ITest :a="props.a"></ITest>
</template>

<script setup>
import Itest from './ITest.vue'
/* eslint-disable */
const props = defineProps({
  a: String
})
</script>
js 复制代码
ITest.vue

<template>
    <div>{{props.a}}</div>
</template>
<script setup>
/* eslint-disable */
 // eslint-disable-next-line 
const props = defineProps({
    a: String
})
</script>
  1. 通过点击按钮触发a值变化查看渲染流程区别

情况 1

情况 2

我们可以观察到情况1 在第一次patch的时候renderfunc这个子组件是没有触发到内部的patch,而是通过updateProps再度调用renderfunc进入内部,从而内部patch的。

部分源码路径

这里我们可以观察到renderfunc这个组件的patchflag为0,shapeflag为2(patch时这两个key启到判断作用),进入processComponent =》 updateComponent,因为新的component为空所以直接赋值旧的就结束了。第一次的patch就这样结束。又由于h里面使用props会触发收集所以在updateprops的时候就会调用renderfunc这个函数,从而触发patch。

总结

通过源码的debug和perfomance的记录我们可以得出使用动态渲染这个写法确实会造成一些性能上的损耗,多次多层的不断触发渲染以及patch会更加明显的看出来。

参考

vue3 源码

相关推荐
大得36926 分钟前
electron结合vue,直接访问静态文件如何跳转访问路径
javascript·vue.js·electron
水银嘻嘻2 小时前
12 web 自动化之基于关键字+数据驱动-反射自动化框架搭建
运维·前端·自动化
小嘟嚷ovo3 小时前
h5,原生html,echarts关系网实现
前端·html·echarts
十一吖i3 小时前
Vue3项目使用ElDrawer后select方法不生效
前端
只可远观3 小时前
Flutter目录结构介绍、入口、Widget、Center组件、Text组件、MaterialApp组件、Scaffold组件
前端·flutter
周胡杰3 小时前
组件导航 (HMRouter)+flutter项目搭建-混合开发+分栏效果
前端·flutter·华为·harmonyos·鸿蒙·鸿蒙系统
敲代码的小吉米4 小时前
前端上传el-upload、原生input本地文件pdf格式(纯前端预览本地文件不走后端接口)
前端·javascript·pdf·状态模式
是千千千熠啊4 小时前
vue使用Fabric和pdfjs完成合同签章及批注
前端·vue.js
九月TTS4 小时前
TTS-Web-Vue系列:组件逻辑分离与模块化重构
前端·vue.js·重构
我是大头鸟5 小时前
SpringMVC 内容协商处理
前端