一、函数式组件和普通组件的区别

  1. 渲染快
  2. 没有实例,意味着没有(this)
  3. 没有生命周期(没有响应式数据)

二、组件函数的使用

  1. 以局部组件为例,将组件标记为 functional=ture;

因为函数式没有实例,因此组件需要的一切都是通过 context 参数传递,它是一个包括如下字段的对象:

    • props:提供所有 prop 的对象
    • children: VNode 子节点的数组
    • slots: 一个函数,返回了包含所有插槽的对象
    • scopedSlots: (2.6.0+) 一个暴露传入的作用域插槽的对象。也以函数形式暴露普通插槽。
    • data:传递给组件的整个数据对象,作为 createElement 的第二个参数传入组件
    • parent:对父组件的引用
    • listeners: (2.3.0+) 一个包含了所有父组件为当前组件注册的事件监听器的对象。这是 data.on 的一个别名。
    • injections: (2.3.0+) 如果使用了inject选项,则该对象包含了应当被注入的属性。

在添加 functional: true 之后,需要更新我们的锚点标题组件的渲染函数,为其增加 context参数,并将 this.$slots.default 更新为 context.children,然后将 this.level 更新为 context.props.level

因为函数式组件只是函数,所以渲染开销也低很多。

在作为包装组件时它们也同样非常有用。比如,当你需要做这些时:

    • 程序化地在多个组件中选择一个来代为渲染;
    • 在将 childrenpropsdata 传递给子组件之前操作它们。
     data() {
         return {
             changer:1
         }
     },
   components: {
         MyCmp:{
             functional:true,   //必要的设置
             render: function (createElement, context) {
                 function getcomp(cmp){
                     console.info(this); //输出为undefined,证明没有实例
                     if(cmp==1){
                         return comp1;
                     }else{
                         return comp2
                     }
                 }
                 return createElement(getcomp(context.props.changer),
                 {
                     props:{
                         cmpData:context.props.data //为子组件传递数据
                     }
                 }
                 );
             },

2. 定义要渲染的组件

 var comp1={
     props:['cmpData'],
     render:function(createElement,context){
         return createElement('el-input',{
             props:{
                 type:this.cmpData
             }
         });
     },
     mounted() {
         console.log(this)  //这个组件为正常组件
     },
 }
 var comp2={
     props:['cmpData'],
     render:function(createElement,context){
         return createElement('el-button',{
             props:{
                 type:this.cmpData
             }
         });
     },
     mounted() {
         console.log(this)  //正常组件
     },
 }

三、在父组件中使用

 <template>
     <div>
         <el-input v-model="changer" placeholder="子组件"></el-input>

         <my-cmp :changer="changer"></my-cmp>
     </div>
 </template>
 <script>

四、理解渲染函数的参数

接下来说一下createElement 接受的参数:

第一个参数:可以是  {String | Object | Function}

不管是那种类型,最终返回到都是需要渲染的普通DOM标签,

第二个参数:是一个对象,这个参数是可选的,定义了需要渲染组件的参数,相对于普通HTML标签的属性是一样的。

还可以自定义指令的,Vue特有的东西,只是抽象一些,没有直接用Vue.directive()用起来直观。

第三个参数:子级虚拟节点,如果你这个节点只是单节点,没有嵌套节点,这个参数可以忽略。如果有的你就要使用一个数据数组的值位cerateElement()返回的虚拟节点。套路都是一样的。

 // @returns {VNode}
 createElement(
   // {String | Object | Function}
   // 一个 HTML 标签名、组件选项对象,或者
   // resolve 了上述任何一种的一个 async 函数。必填项。
   'div',

   // {Object}
   // 一个与模板中属性对应的数据对象。可选。
   {
     // 与 `v-bind:class` 的 API 相同,
     // 接受一个字符串、对象或字符串和对象组成的数组
     'class': {
         foo: true,
         bar: false
     },
     // 与 `v-bind:style` 的 API 相同,
     // 接受一个字符串、对象,或对象组成的数组
     style: {
         color: 'red',
         fontSize: '14px'
     },
     // 普通的 HTML 特性
     attrs: {
         id: 'foo'
     },
     // 组件 prop
     props: {
         myProp: 'bar'
     },
     // DOM 属性
     domProps: {
         innerHTML: 'baz'
     },
     // 事件监听器在 `on` 属性内,
     // 但不再支持如 `v-on:keyup.enter` 这样的修饰器。
     // 需要在处理函数中手动检查 keyCode。
     on: {
         click: this.clickHandler
     },
     // 仅用于组件,用于监听原生事件,而不是组件内部使用
     // `vm.$emit` 触发的事件。
     nativeOn: {
         click: this.nativeClickHandler
   },
   // 自定义指令。注意,你无法对 `binding` 中的 `oldValue`
   // 赋值,因为 Vue 已经自动为你进行了同步。
   directives: [
     {
       name: 'my-custom-directive',
       value: '2',
       expression: '1 + 1',
       arg: 'foo',
       modifiers: {
         bar: true
       }
     }
   ],
   // 作用域插槽的格式为
   // { name: props => VNode | Array<VNode> }
   scopedSlots: {
     default: props => createElement('span', props.text)
   },
   // 如果组件是其它组件的子组件,需为插槽指定名称
   slot: 'name-of-slot',
   // 其它特殊顶层属性
   key: 'myKey',
   ref: 'myRef',
   // 如果你在渲染函数中给多个元素都应用了相同的 ref 名,
   // 那么 `$refs.myRef` 会变成一个数组。
   refInFor: true
   },

   // {String | Array}
   // 子级虚拟节点 (VNodes),由 `createElement()` 构建而成,
   // 也可以使用字符串来生成“文本虚拟节点”。可选。
   [
     '先写一些文字',
     createElement('h1', '一则头条'),
     createElement(MyComponent, {
       props: {
         someProp: 'foobar'
       }
     })
   ]
 )

Vue函数式组件的应用的更多相关文章

  1. Vue 函数式组件 functional

    函数式组件 无状态 无法实例化 内部没有任何生命周期处理函数 轻量,渲染性能高,适合只依赖于外部数据传递而变化的组件(展示组件,无逻辑和状态修改) 在template标签里标明functional 只 ...

  2. vue函数式组件详解

    本篇将详细介绍vue组件化之函数式组件,会用到以下api: Vue.component().Vue.extend().$createElement.patch(). 从事vue开发的小伙伴,平时组件化 ...

  3. Vue.js 源码分析(三十) 高级应用 函数式组件 详解

    函数式组件比较特殊,也非常的灵活,它可以根据传入该组件的内容动态的渲染成任意想要的节点,在一些比较复杂的高级组件里用到,比如Vue-router里的<router-view>组件就是一个函 ...

  4. 【vue】函数式组件

    在 2.5.0 及以上版本中,如果你使用了单文件组件,那么基于模板的函数式组件可以这样声明: <template functional> <div class="cell& ...

  5. 渲染函数render和函数式组件

    vnode对象 vnode对象包括(vnode并不是vue实例,而是vue实例中渲染函数render执行后生成的结果) this.tag = tag // 当前节点标签名 this.data = da ...

  6. Vuejs函数式组件,你值得拥有(1)

    函数式组件在React社区很流行使用,那么在vue里面我们要怎么用呢 下面会涉及到的知识点: 高阶函数.状态.实例.vue组件 什么是函数式组件 我们可以把函数式组件想像成组件里的一个函数,入参是渲染 ...

  7. vue.js组件化开发实践

    前言 公司目前制作一个H5活动,特别是有一定统一结构的活动,都要码一个重复的轮子.后来接到一个基于模板的活动设计系统的需求,便有了下面的内容.借油开车. 组件化 需求一到,接就是怎么实现,技术选型自然 ...

  8. 如何理解vue.js组件的作用域是独立的

    vue.js组件的作用域是独立,可以从以下三个方面理解: 1.父组件模板在父组件作用域内编译,父组件模板的数据用父组件内data数据:2.子组件模板在子组件作用域内编译,子组件模板的数据用子组件内da ...

  9. Vue 子组件向父组件传参

    直接上代码 <body> <div id="counter-event-example"> <p>{{ total }}</p> & ...

随机推荐

  1. Linux版本划分——基于打包方式

    基于Dpkg (Debian系) Debian GNU / Linux是一种强调使用自由软件的发行版.它支持多种硬件平台.Debian及其派生发行版使用deb软件包格式,并使用dpkg及其前端作为包管 ...

  2. 【SVN】SVN使用教程总结

    SVN使用教程总结 SVN简介: 为什么要使用SVN? 程序员在编写程序的过程中,每个程序员都会生成很多不同的版本,这就需要程序员有效的管理代码,在需要的时候可以迅速,准确取出相应的版本. Subve ...

  3. linux初学者-DDNS配置篇

    linux初学者-DDNS配置篇 如果DNS服务器要记录多台主机的IP,且这些主机的IP都是通过DHCPD服务自动获取的,那么将会造成很大的困难,因为在DNS设置时无法得知主机具体的IP.如果DHCP ...

  4. 必懂的webpack高级配置

    webpack高级配置 1.HTML中img标签的图片资源处理 使用时.只需要在html中正常引用图片即可.webpack就会找到对应的资源进行打包.并修改html中的引用路径 主要是将html中的i ...

  5. 备战金九银十,Java研发面试题(Spring、MySQL、JVM、Mybatis、Redis、Tomcat)[带答案],刷起来!

    八月在即,马上就是"金九银十",又是跳槽招聘季.咱们这行公认涨薪不如跳槽加的快.但不建议频繁跳槽,还是要学会融合团队,抓住每个机会提升技能. 苏先生在这里给大家整理了一套各大互联网 ...

  6. Android的步骤

    1.开发Activity步骤 第一步:写一个类继承Activity 第二步:重写oncreate方法 第三步:在主配置文件中注册activity <activity android:name=& ...

  7. java 第三章

       if  选择结构:        语法:if(条件){ //代码块1 } if——else选择结构 语法:if(条件){ //代码块1 }else{ //代码块2 } 多重if选择结构 语法:i ...

  8. GStreamer基础教程06 - 获取媒体信息

    摘要 在常见的媒体文件中,通常包含一些数据(例如:歌手,专辑,编码类型等),用于描述媒体文件.通常称这些数据为元数据(Metadata:data that provides information a ...

  9. 【MySQL】ON DUPLICATE KEY UPDATE

    之前没用过这个操作,甚至没见过--最近接触到,而且还挺有用. 作用:若 KEY 不重复,则插入记录:否则更新记录. 单条操作: INSERT INTO table(a, b, c) VALUES (1 ...

  10. postman使用pre-request script计算md5

    接口加了验签逻辑,具体是md5(salt+时间戳).被某君吐槽说测试不方便啊能不能先关掉.其实没有必要打开又关闭验签功能,postman的pre-request script功能完全可以模拟客户端加密 ...