Vue | Vue深入浅出——Vue中的render函数详解

news2024/11/15 11:03:01

1.render函数

在编写vue单文件的大多数情况下,我们都是使用template模板来创建HTML。然而在一些条件判断比较复杂的场景下,使用JavaScript去描绘HTML的生成逻辑会显得更加的简洁直观。

使用Vue官网的例子来简单说明:

如果自己在开发的时候,编写的每个标题(包括h1~h6)都需要带锚点,如下所示:

<h1>
  <a name="hello-world" href="#hello-world">
    Hello world!
  </a>
</h1>

如果用template模板进行编写,会如下所示:

<template>
  <h1 v-if="level === 1">
    <anchor :name="name" :content="content"></anchor>
  </h1>
  <h2 v-else-if="level === 2">
    <anchor :name="name" :content="content"></anchor>
  </h2>
  <h3 v-else-if="level === 3">
    <anchor :name="name" :content="content"></anchor>
  </h3>
  <h4 v-else-if="level === 4">
    <anchor :name="name" :content="content"></anchor>
  </h4>
  <h5 v-else-if="level === 5">
    <anchor :name="name" :content="content"></anchor>
  </h5>
  <h6 v-else-if="level === 6">
    <anchor :name="name" :content="content"></anchor>
  </h6>
</template>
<script>
export default {
  name:'anchor-header',
  props:{
    level:Number,
    name:String,
    content:String
  },
  components:{
    'anchor':{
      props:{
        content:String,
        name:String
      },
      template:'<a :id="name" :href="`#${name}`"> {{content}}</a>'
    }
  }
}
</script>

显然代码冗长累赘。但如果用render函数来编写则如下所示:

<script>
export default {
  name:'anchor-header',
  props:{
    level:Number,
    name:String,
    content:String
  },
  render:function(createElement){
    const anchor={
      props:{
        content:String,
        name:String
      },
      template:'<a :id="name" :href="`#${name}`"> {{content}}</a>'
    }
    const anchorEl=createElement(anchor,{
      props:{
        content:this.content,
        name:this.name
      }
    })
    const el=createElement(
      `h${this.level}`,
      [anchorEl]
    )
    return el
  }
}
</script>

可见通过render函数编写出的逻辑更加简洁且可读性更高。

每一个render函数都要return一个VNode类型的变量,是Vue中自定义的虚拟节点(virtual node),用于替换挂载元素$el。

Vue 选项中的 render 函数若存在,则 Vue 构造函数不会从 template 选项或通过 el 选项指定的挂载元素中提取出的 HTML 模板编译渲染函数。

但自己实践在vue单文件实践后发现,如果同时存在templaterender,生成的html会以template的逻辑为主,奇怪。

上述例子的代码可以知道,render函数使用的场景是,要根据不同条件切换不同HTML标签,可以使用render函数。或者条件判断较多的template中,用渲染函数编写会让代码的可读性更高的情况下,也推荐使用render函数。

下面分析上面代码中出现的createElement函数。

2.createElement

createElement用于创建且return一个VNode类型的变量(虚拟节点),以下是该函数的传入参数:

(1)一个 HTML 标签名、组件选项对象或者resolve前面两种之一类型的async函数。必填。

例如,传入'div'表示想创建一个html标签为div的VNode;如果传入'transition'代表创建transition组件。

上面作为粒子的代码就是根据传入的类型为Number的参数level,用模板字符串拼接成标签名称。传入level为1则拼接出来的HTML标签名称为h1。代表要创建html标签为h1的VNode。

(2)所创建的VNode中所需参数为属性的数据对象。可选。

{
  class,
  style,
  attrs,
  props,
  domProps,
  on,
  nativeOn,
  directives:[
    {
      name,
      value,
      expression,
      arg,
      modifiers
    }
  ],
  scopedSlots:{
    default:props=>createElement()
  },
  slot,
  key,
  ref,
  refInFor
}

大部分属性和vue组件中存在的属性的作用一样,我就只挑几个比较特殊的属性来说明:

**nativeOn:**用于监听原生事件,而不是组件内使用。例如:

nativeOn: {click: this.nativeClickHandler}

相当于@click.native="nativeClickHandler"

scopedSlots:定义作用域插槽的内部的内容:

格式为:{ name: props => VNode | Array<VNode> }

举一个例子:

<script>
export default {
  render (createElement) {
    var component = {
      template: `<div>
        <slot></slot>
        <slot name="foo"></slot>
      </div>`
    }
    return createElement(component, {
      scopedSlots: {
        default: props => createElement('span', '456'),
        foo: props => createElement('span', '789')
      },
    })
  }
}
</script>

最后渲染出来的html效果如下:

<div>
     <span>456</span>
     <span>789</span>
</div>

scope:如果要生成的组件要插入到,需为插槽指定名称。

举个例子

<script>
export default {
  render (createElement) {
    var component = {
      template: `<div>
        <slot></slot>
        <slot name="foo"></slot>
      </div>`
    }
    const childrenEl = createElement('span', { slot: 'foo' }, '123')
    return createElement(component, {
      scopedSlots: {
        default: props => createElement('span', '456')
        // foo: props => createElement('span', '789')      },
    }, [childrenEl])
  }
}
</script>

最后渲染出来的html效果如下:

<div>
     <span>456</span>
     <span>123</span>
</div>

注意:如果去掉上面例子的代码中// foo: props => createElement('span', '789')的注释,则slot="foo"插槽中显示内容依然为<span>789</span>

**  refInFor:**如果你在渲染函数中给多个元素都应用了相同的 ref 名,那么 `$refs.myRef` 会变成一个数组。

(3)子级虚拟节点 (VNodes)。如果传入的是VNode则要用列Array传入,另外也可以使用字符串来生成“文本虚拟节点”。可选。

3.函数式组件

函数式组件相比于一般的vue组件而言,最大的区别是非响应式的它不会监听任何数据,也没有实例(因此没有状态,意味着不存在诸如created,mounted的生命周期)。好处是因只是函数,故渲染开销也低很多。

把开头的例子改成函数式组件,代码如下:

<script>
export default {
  name:'anchor-header',
  functional:true, // 以functional:true声明该组件为函数式组件
  props:{
    level:Number,
    name:String,
    content:String
  },
  // 对于函数式组件,render函数会额外传入一个context参数用来表示上下文,即替代this。函数式组件没有实例,故不存在this
  render:function(createElement,context){ 
    const anchor={
      props:{
        content:String,
        name:String
      },
      template:'<a :id="name" :href="`#${name}`"> {{content}}</a>'
    }
    const anchorEl=createElement(anchor,{
      props:{
        content:context.props.content, //通过context.props调用props传入的变量
        name:context.props.name
      }
    })
    const el=createElement(
      `h${context.props.level}`,
      [anchorEl]
    )
    return el
  }
}
</script>

渲染函数 & JSX — Vue.js:更多关于函数式组件内容请看官网函数式组件

4.element-ui的el-row组件

最后以el-row组件的源码来分析,该源码的渲染逻辑在render函数上,非常简洁明了:

export default {
  name: 'ElRow',

  componentName: 'ElRow',

  props: {
    tag: {
      type: String,
      default: 'div'
    },
    gutter: Number,
    type: String,
    justify: {
      type: String,
      default: 'start'
    },
    align: {
      type: String,
      default: 'top'
    }
  },

  computed: {
    style() {
      const ret = {};

      if (this.gutter) {
        ret.marginLeft = `-${this.gutter / 2}px`;
        ret.marginRight = ret.marginLeft;
      }

      return ret;
    }
  },

  render(h) {
    return h(this.tag, {
      class: [
        'el-row',
        this.justify !== 'start' ? `is-justify-${this.justify}` : '',
        this.align !== 'top' ? `is-align-${this.align}` : '',
        { 'el-row--flex': this.type === 'flex' }
      ],
      style: this.style
    }, this.$slots.default);
  }
};

对着el-row组件传入参数的说明图来解释:

直接从render函数处进行分析,

1.传入第一个参数为this.tag,用于根据参数生成对应的html标签。

2.第二个参数中传入classstyle是根据props中的type,gutter,justify,align生成的。

3.第三个参数传入子节点。此处通过this.$slots.default拿到传入的子节点。例如:

<el-row>
    <div>123</div>
</el-row>

此时,this.$slots.default 获取的数据则是一个包含上面<div>123</div>的VNode的数组。

以下内容来自官网:

拓展:slots() 和 children 对比

你可能想知道为什么同时需要 slots() childrenslots().default 不是和 children 类似的吗?在一些场景中,是这样——但如果是如下的带有子节点的函数式组件呢?

<my-functional-component>
  <p v-slot:foo>
    first
  </p>
  <p>second</p>
</my-functional-component>

对于这个组件,children 会给你两个段落标签,而 slots().default 只会传递第二个匿名段落标签,slots().foo 会传递第一个具名段落标签。同时拥有 children slots(),因此你可以选择让组件感知某个插槽机制,还是简单地通过传递 children,移交给其它组件去处理。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vscode配置django环境并创建django项目

1、创建文件夹 创建文件夹 并在vscode打开 终端输入命令 “ python -m venv env ” 查看目录结构 2、创建项目 在终端输入 django-admin startproject 文件名(这里以myshop为例) 3、创建应用 在myshop打开终端 在终端输入 django-admin startapp 应用名 这里以app1为例…

6个免费icon图标素材网站

在这个数字化时代&#xff0c;优秀的图标设计对于提升用户体验至关重要。为了帮助设计师和开发者找到高质量的免费icon图标素材&#xff0c;我整理了以下6个实用的网站&#xff0c;让你轻松获取精美图标&#xff0c;助力你的设计工作。快来一起看看吧&#xff01; 1、菜鸟图库 …

如何显示Dialog窗口

文章目录 1. 概念介绍2. 使用方法2.1 Overlay效果2.1 Dialog效果 3. 示例代码4. 内容总结 我们在上一章回中介绍了"使用get显示snackBar"相关的内容&#xff0c;本章回中将介绍使用get显示Dialog.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在…

【卷起来】VUE3.0教程-07-异步请求处理(springboot后端)

&#x1f332; 服务端接口准备 pom文件&#xff0c;引入mybatis/mybatis-plus相关依赖 <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency>&…

Split函数

Split:可以将一个完整的字符串按照指定的分隔符划分为若干个子字符串。 String[] split(String regex) :将字符串全部拆分. String[] split(String regex, int limit) :将字符串以指定的格式&#xff0c;拆分为limit 组 代码实例&#xff1a; 切割小数点&#xff0c;或\ ,需…

每日单词记背

2024年9月12日 1.discriminate&#xff1a;歧视&#xff0c;区别&#xff0c;分辨 discriminate against 歧视&#xff1b;排斥 discriminate between 区别 辨别 dis(区别)crim(罪犯)inate ->区别为罪犯->歧视 it is illegal to discriminate against women in any way.…

【Linux修行路】信号的产生

目录 ⛳️推荐 一、信号的产生 二、产生信号的系统调用 2.1 kill——给指定的进程发送指定的信号 2.2 模拟实现指令 kill 2.3 raise——给调用的进程发送指定的信号 2.4 abort——给调用者发送 6 号信号 三、验证哪些信号不可以被捕捉 四、为什么除0和解引用空指针会给…

数据库(DB、DBMS、SQL)

今天我来讲解一下数据库和可视化数据库管理系统的使用 数据库概述 数据库 存储数据的仓库&#xff0c;数据是有组织的存储 DataBase (DB) 数据库管理系统 操纵和管理数据库的大型软件 DataBaseMangement System (DBMS) SQL 操作关系型数据库的编程语言&#xff0c;定义…

探索最佳 Shell 工具:全面测评 Bash、Zsh、Fish、Tcsh 和 Ksh

感谢浪浪云支持发布 浪浪云活动链接 &#xff1a;https://langlangy.cn/?i8afa52 文章目录 1. 简介2. 测评工具3. 测评标准4. Bash 测评4.1 易用性4.2 功能特性4.3 性能4.4 可定制性4.5 社区和支持 5. Zsh 测评5.1 易用性5.2 功能特性5.3 性能5.4 可定制性5.5 社区和支持 6. F…

C++设计模式——Builder Pattern建造者模式

一&#xff0c;建造者模式的定义 建造者模式&#xff0c;又被称为生成器模式&#xff0c;是一种创建型设计模式&#xff0c;它将复杂产品的构建过程分解为一系列简单的步骤&#xff0c;每个步骤由独立的建造者对象负责。 建造者模式常用于创建复杂的对象&#xff0c;它避免了…

网络安全架构师

网络安全架构师负责构建全面的安全框架&#xff0c;以保护组织的数字资产免受侵害&#xff0c;确保组织在数字化转型的同时维持强大的安全防护。 摩根大通的网络安全运营副总裁兼安全架构总监Lester Nichols强调&#xff0c;成为网络安全架构师对现代企业至关重要&#xff0c;…

单向链表之创建,插入,输出(上)

文章目录 &#x1f34a;自我介绍&#x1f34a;创建&#x1f34a;插入&#x1f34a;输出 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以&#xff1a;点赞关注评论收藏&#xff08;一键四连&#xff09;哦~ &#x1f34a;自我介绍 Hello,大家好&#xff0c;我是小珑也要…

VMware Fusion虚拟机Mac版 安装Ubuntu操作系统教程

Mac分享吧 文章目录 下载镜像地址&#xff1a;[www.macfxb.cn](http://www.macfxb.cn)一、CentOS安装完成&#xff0c;软件打开效果二、Mac中安装Ubuntu虚拟机1️⃣&#xff1a;下载镜像2️⃣&#xff1a;创建虚拟机3️⃣&#xff1a;虚拟机设置4️⃣&#xff1a;虚拟机安装5️…

计算机三级 - 数据库技术 - 第十四章 数据仓库与数据挖掘 笔记

第十四章 数据仓库与数据挖掘 内容提要&#xff1a; 了解数据仓库相关技术了解数据仓库的设计、建造、运行及维护了解OLAP及多维数据模型了解数据挖掘技术 决策支持系统(DSS)&#xff1a;综合利用大量数据有机组合众多模型(数学模型和数据处理模型)&#xff0c;通过人机交互&a…

uniapp 端开发 echarts 树结构图

实现效果 &#xff1a; 1. 在uniapp 中写echarts 树结构图需要使用 <script module"echarts" lang"renderjs"> 否则会无法显示echarts 图形 rebderjs 代码 引入了 /static/echarts.min.js 是在 ECharts 在线构建 定制你的echarts <te…

001 RabbitMQ入门及安装

RabbitMQ入门及安装 文章目录 RabbitMQ入门及安装1.介绍1.AMQP和JMS2.目前主流的消息队列 2.安装1.Linux安装1.1 安装erlang1.2 RabbitMQ安装 2.Docker安装 3.核心组件 1.介绍 RabbitMQ是实现了高级消息队列协议&#xff08;AMQP&#xff09;的开源消息代理软件&#xff08;亦…

嵌入式音视频开发:探索多领域的融合与创新

摘要&#xff1a; 本文深入探讨了嵌入式音视频开发领域。从嵌入式系统的基础概念入手&#xff0c;阐述了其在音视频领域的独特地位。详细介绍了嵌入式音视频开发中涉及的硬件组件&#xff0c;如处理器、编解码器、存储设备等。分析了音视频编解码技术&#xff0c;包括常见的编解…

空间数据库概述

空间数据库简介 空间数据库是 地理信息系统 在计算机物理存储介质中存储的&#xff0c;与GIS应用相关的地理空间数据的总和。一般以一系列特定结构的文件形式组织后存储在介质上。 空间数据库的特点 可以存储、处理空间数据相比普通数据库提供更多、更复杂的数据类型以及更多…

[SWPU2019]Web1 超详细教程

老规矩先看源码&#xff0c;没找到啥提示&#xff0c;后面就是登录口对抗 弱口令试了几个不行&#xff0c;就注册了个账户登录进去 可以发布广告&#xff0c;能造成xss&#xff0c;但是没啥用啊感觉 查看广告信息的时候&#xff0c;注意到url当中存在id参数&#xff0c;可能存…

Leetcode面试经典150题-134.加油站

解法都在代码里&#xff0c;不懂就留言或者私信 class Solution {public int canCompleteCircuit(int[] gas, int[] cost) {/**如果只有一个加油站&#xff0c;那它本来就在那个为止&#xff0c;0就是它的编号?但是这只是你的想象&#xff0c;题目有个变态规定&#xff0c;自…