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渲染函数的子节点传递,需要以"插槽函数"的形式进行传递++。

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

相关推荐
kiramario8 分钟前
【结束】JS如何不通过input的onInputFileChange使用本地mp4文件并播放,nextjs下放入public文件的视频用video标签无法打开
开发语言·javascript·音视频
小盼江1 小时前
水果生鲜农产品推荐系统 协同过滤余弦函数推荐水果生鲜农产品 Springboot Vue Element-UI前后端分离 代码+开发文档+视频教程
vue.js·spring boot·ui
化作繁星2 小时前
如何在 React 中测试高阶组件?
前端·javascript·react.js
初遇你时动了情2 小时前
react module.scss 避免全局冲突类似vue中scoped
vue.js·react.js·scss
Au_ust2 小时前
千峰React:函数组件使用(2)
前端·javascript·react.js
烂蜻蜓2 小时前
Uniapp 设计思路全分享
前端·css·vue.js·uni-app·html
GAMESLI-GIS2 小时前
【WebGL】fbo双pass案例
前端·javascript·webgl
bin91533 小时前
DeepSeek 助力 Vue 开发:打造丝滑的二维码生成(QR Code)
前端·javascript·vue.js·ecmascript·deepseek
@LitterFisher3 小时前
Excell 代码处理
前端·javascript·excel
十八朵郁金香4 小时前
【JavaScript】深入理解模块化
开发语言·javascript·ecmascript