vue 函数化组件

简介: 【10月更文挑战第1天】 Vue.js 的函数化组件通过设置 `functional: true`,使其无状态和无实例,从而减少渲染开销。通过 `render` 函数的 `context` 参数传递数据。示例中,`smart-item` 组件根据 `data.type` 动态选择并渲染 `ImgItem`、`VideoItem` 或 `TextItem` 组件。根实例 `app` 通过按钮切换不同类型的组件数据。函数化组件适用于程序化选择组件和操作传递数据的场景。

Vue.js提供了一个functional的布尔值选项,设置为true可以使组件无状态和无实例,也就是没有data和this上下文。这样用render函数返回虚拟节点可以更容易渲染,因为函数化组件只是一个函数,渲染开销要小很多。

使用函数化组件时,Render函数提供了第二个参数context来提供临时上下文。组件需要的data、props、slots、children、parent都是通过这个上下文来传递的,比如this.level要改写为context.props.level, this.$slots.default改写为context.children。

例如,下面的示例用函数化组件展示了一个根据数据智能选择不同组件的场景:

<div id="app">
            <smart-item :data="data"></smart-item>
            <button @click="change('img')">切换为图片组件</button>
            <button @click="change('video')">切换为视频组件</button>
            <button @click="change('text')">切换为文本组件</button>
        </div>
        <script>
            // 图片组件选项
            var ImgItem = {
   
              props: ['data'],
              render: function (createElement) {
   
                  return createElement('div', [
                      createElement('p', ’图片组件’),
                      createElement('img', {
   
                          attrs: {
   
                            src: this.data.url
                          }
                      })
                  ]);
              }
            };
            // 视频组件选项
            var VideoItem = {
   
              props: ['data'],
              render: function (createElement) {
   
                  return createElement('div', [
                      createElement('p', ’视频组件’),
                      createElement('video', {
   
                          attrs: {
   
                            src: this.data.url,
                            controls: 'controls',
                            autoplay: 'autoplay'
                          }
                      })
                  ]);
              }
            };
            // 纯文本组件选项
            var TextItem = {
   
                props: ['data'],
                render: function (createElement) {
   
                  return createElement('div', [
                      createElement('p', ’纯文本组件’),
                      createElement('p', this.data.text)
                  ]);
                }
            };
            Vue.component('smart-item', {
   
                //函数化组件
                functional: true,
                render: function (createElement, context) {
   
                  // 根据传入的数据,智能判断显示哪种组件
                  function getComponent () {
   
                      var data = context.props.data;
                      // 判断prop: data的type字段是属于哪种类型的组件
                      if (data.type === 'img')   return ImgItem;
                      if (data.type === 'video') return VideoItem;
                      return TextItem;
                  }
                  return createElement(
                      getComponent(),
                      {
   
                          props: {
   
                              //把smart-item的prop: data传给上面智能选择的组件
                              data: context.props.data
                          }
                      },
                      context.children
                  )
                },
                props: {
   
                  data: {
   
                      type: Object,
                      required: true
                  }
                }
            })
            var app = new Vue({
   
                el: '#app',
                data: {
   
                  data: {
   }
                },
                methods: {
   
                    // 切换不同类型组件的数据
                    change: function (type) {
   
                       if (type === 'img') {
   
                       this.data = {
   
                          type: 'img',
                          url: 'https://raw.githubusercontent.com/iview/iview/master/assets/logo.png'
                          }
                       } else if (type === 'video') {
   
                          this.data = {
   
                             type: 'video',
                             url: 'http://vjs.zencdn.net/v/oceans.mp4">http://vjs.zencdn.net/v/oceans.mp4'
                          }
                       } else if (type === 'text') {
   
                          this.data = {
   
                             type: 'text',
                             content: ’这是一段纯文本’
                          }
                       }
                    }
                 },
                 created: function () {
   
                    // 初始化时,默认设置图片组件的数据
                    this.change('img');
               }
           })
        </script>

代码片段比较长,逐步分析一下实现的内容。ImgItem、VideoItem、TextItem这3个对象分别是图片组件、视频组件和纯文本组件的选项,它们都接收一个prop:data。在函数化组件smart-item里,也有props:data,通过getComponent函数来判断其字段type的值,选择这条数据适合渲染的组件。通过createElement把getComponent()返回的对象设置为第一个参数,然后通过第二个参数把smart-item的data传递到选择的组件里的prop:data,组件渲染出不同的内容。

根实例app中的方法change用来生成不同的数据,通过3个button来切换。

该示例难理解的地方在于smart-item和3个功能组件都有prop:data,它们的传递顺序和原理看起来比较含糊。

函数化组件在业务中并不是很常用,而且也有其他类似的方法来实现,比如上例也可以用组件的is特性来动态挂载。总结起来,函数化组件主要适用于以下两个场景:

● 程序化地在多个组件中选择一个。

● 在将children, props,data传递给子组件之前操作它们。

相关文章
|
1天前
|
JavaScript 开发者
vue指令的开发看这篇文章就够了!超详细,赶快收藏!
【10月更文挑战第8天】vue指令的开发看这篇文章就够了!超详细,赶快收藏!
vue指令的开发看这篇文章就够了!超详细,赶快收藏!
|
2天前
|
JavaScript 前端开发
Vue开发必备:$nextTick方法的理解与实战场景
Vue开发必备:$nextTick方法的理解与实战场景
23 1
|
2天前
|
JavaScript 前端开发
VUE学习三:双向绑定指令(v-mode)、组件化开发(全局组件/局部组卷/组件通信)、组件化高级(slot插槽使用)
这篇文章是关于Vue.js框架中的v-model指令和组件化开发的详细教程,涵盖了从基础使用到高级功能的多个方面。
17 1
|
2天前
|
JavaScript
Vue路由传参实战指南:三种常用方法示例演示
Vue路由传参实战指南:三种常用方法示例演示
10 0
|
2天前
|
JavaScript
Vue启动时报错的解决方案,以及解决相同路径跳转报错的问题
Vue启动时报错的解决方案,以及解决相同路径跳转报错的问题
20 0
|
3天前
|
JavaScript 数据可视化
vue-cli学习一:vue脚手架的 vue-cli2和vue-cli3版本 创建vue项目,vue的初始化详解
这篇文章介绍了如何使用vue-cli 2和3版本来创建Vue项目,并详细说明了两者之间的主要区别。
23 5
vue-cli学习一:vue脚手架的 vue-cli2和vue-cli3版本 创建vue项目,vue的初始化详解
|
4天前
|
JavaScript
vue知识点
10月更文挑战第2天
20 3
|
2天前
|
存储 JavaScript 前端开发
Vue中组件通信方式有哪些?
本文介绍了 Vue 中几种常见的组件间通信方式,包括 Props / $emit、provide / inject、ref / refs、eventBus、Vuex、$parent / $children、$attrs / $listeners 以及通过 vue-router 传参。每种方式都有其适用场景和注意事项,帮助开发者根据具体需求选择合适的通信方式。
10 3
Vue中组件通信方式有哪些?
|
2天前
|
JavaScript
VUE学习二:事件监听(v-on)、条件判断(v-if/v-else-if/v-else)、循环遍历(v-for)
这篇文章是关于Vue.js的学习指南,涵盖了事件监听、条件判断、循环遍历、数组方法响应性、案例分析以及高阶函数的使用。
14 2
VUE学习二:事件监听(v-on)、条件判断(v-if/v-else-if/v-else)、循环遍历(v-for)
|
3天前
|
JavaScript 数据可视化
vue-cli学习二:vue-cli3版本 创建vue项目后,Runtime-Compiler和Runtime-only两个模式详解;vue项目管理器;配置文件的配置在哪,以及如何配置
这篇文章详细介绍了Vue CLI 3版本创建项目时的Runtime-Compiler和Runtime-only两种模式的区别、Vue程序的运行过程、render函数的使用、eslint的关闭方法,以及Vue CLI 2和3版本配置文件的不同和脚手架3版本创建项目的配置文件配置方法。
13 3
vue-cli学习二:vue-cli3版本 创建vue项目后,Runtime-Compiler和Runtime-only两个模式详解;vue项目管理器;配置文件的配置在哪,以及如何配置