Vue:h渲染函数性能警告[Non-function value encountered for default slot.]

问题描述

在使用Ant-Design-Vue组件库提供的Table表格组件时,通过columns做列配置,尝试借助customRender渲染枚举类型的字段值时,浏览器开发者工具控制台打印如下警告信息:++[Vue warn]: Non-function value encountered for default slot. Prefer function slots for better performance.++,
vue性能警告信息

问题分析

既然是Vue爆出的警告,我们就把排查的第一个点放在h渲染函数这里。

项目代码中,原始配置方式如下(注意h渲染函数的第三个参数),其中:

  • AntTag:是Ant-Design-Vue组件库提供的Tag标签组件
  • StatusEnum:是自定义的ts枚举类型

h渲染函数

Vue官网对于h渲染函数的完整参数签名如下,

javascript 复制代码
// 完整参数签名
function h(
  type: string | Component,
  props?: object | null,
  children?: Children | Slot | Slots
): VNode

// 省略 props
function h(type: string | Component, children?: Children | Slot): VNode

type Children = string | number | boolean | VNode | null | Children[]

type Slot = () => Children

type Slots = { [name: string]: Slot }

对h渲染函数的参数解释如下,

第一个参数既可以是一个字符串 (用于原生元素) 也可以是一个 Vue 组件定义

第二个参数是要传递的 prop

第三个参数是子节点

结合以上解释,至少可以得到以下3个结论:

  1. 第一个参数此处传递的是Vue组件------没有问题;
  2. 第二个参数此处传递的是Vue组件的prop参数,结合Ant-Degisn-Vue Tag标签API来看------没有问题;
  3. 第三个参数此处传递的是字符串类型的变量值,而官网文档给出的是"子节点"类型------如何定义"子节点"类型???是不是这个地方出现了问题呢?

Vue官网也给出了对应的解释:

当创建一个组件的 vnode 时,子节点必须以插槽函数进行传递。如果组件只有默认槽,可以使用单个插槽函数进行传递。否则,必须以插槽函数的对象形式来传递。

为了方便阅读,当子节点不是插槽对象时,可以省略 prop 参数。

结合官网给出的如下所示的第3个例子(省略了prop参数),很显然,我们这里的第三个字符串类型参数,看起来并不符合"插槽函数"类型。

javascript 复制代码
import Foo from './Foo.vue'

// 传递 prop
h(Foo, {
  // 等价于 some-prop="hello"
  someProp: 'hello',
  // 等价于 @update="() => {}"
  onUpdate: () => {}
})

// 传递单个默认插槽
h(Foo, () => 'default slot')

// 传递具名插槽
// 注意,需要使用 `null` 来避免
// 插槽对象被当作是 prop
h(MyComponent, null, {
  default: () => 'default slot',
  foo: () => h('div', 'foo'),
  bar: () => [h('span', 'one'), h('span', 'two')]
})

插槽函数

那么,什么是插槽函数?参考Vue官网-传递插槽。给出定义:

插槽函数的返回值同一个正常的渲染函数的返回值一样------并且在子组件中被访问时总是会被转化为一个 vnodes 数组

渲染函数

又回到渲染函数这个话题了,那么Vue官网是如何定义"渲染函数"的呢?

详细可参见:Vue官网-渲染机制部分

解决方案

明确了问题的原因:++h渲染函数的子节点传递,需要以"插槽函数"的形式进行传递++。

问题就好解决了,以下是修改之后的项目代码(注意红框部分),问题解决!
修改后的代码

相关推荐
devincob2 小时前
js原生、vue导出、react导出、axios ( post请求方式)跨平台导出下载四种方式的demo
javascript·vue.js·react.js
编程社区管理员2 小时前
React 发送短信验证码和验证码校验功能组件
前端·javascript·react.js
葡萄城技术团队2 小时前
迎接下一代 React 框架:Next.js 16 核心能力解读
javascript·spring·react.js
全马必破三2 小时前
React“组件即函数”
前端·javascript·react.js
課代表2 小时前
JavaScript 中获取二维数组最大值
javascript·max·数组·递归·array·最大值·二维
rising start3 小时前
五、CSS盒子模型(下)
前端·javascript·css
木头没有瓜4 小时前
在 Windows 中清理依赖node_modules并重新安装
vue.js
不吃香菜的猪4 小时前
el-upload实现文件上传预览
前端·javascript·vue.js
dcloud_jibinbin5 小时前
【uniapp】小程序体积优化,分包异步化
前端·vue.js·webpack·性能优化·微信小程序·uni-app
qq_427506085 小时前
基于Vue 3和Element Plus实现简单的钩子函数管理各类弹窗操作
前端·javascript·vue.js