Vue 组件继承在项目中的应用案例

平时在项目中很少用到Vue的继承特性,通常是封装单文件组件,然后在其他组件中引入。继承是面向对象程序设计的一个核心概念之一,通过它可以实现子类对父类的数据和功能的复用,同时还可以选择性地覆写父类的属性和方法。那么,前端组件开发如何利用这个特性呢?

最近刚好碰到一个应用场景,某个业务模块的各个页面在结构上几乎完全一致,区别只在于里面的数据。首先想到的实现方式是封装成一个组件,然后在各个页面中引入,通过props传递不同的数据来区分不同行为。但这种方式需要在组件里做各种判断,每多一个页面类型就需要增加一个条件分支,这样就违背了面向对象设计里的“开发-封闭”原则,不利于以后的扩展。

页面结构

该业务模块是要展示多个统计指标的趋势,页面包含一个走势图和报表,以及一些条件筛选控件。每个指标的页面几乎都是一样的。使用继承,刚好可以解决这个问题。定义一个基类页面,包含页面模板和公共属性和方法,各数据指标页面作为子类继承该基类,这样就拥有了同样的页面模板和事件绑定,只需要在子类覆写事件处理方法就能实现不同统计指标展示不同数据。

Vue 支持组件的继承,在单文件组件里指定extends即可:

<!-- 某统计指标报表 -->
<script>
import ReportBase from './ReportBase.vue';
export default {
  extends: ReportBase,
  data() {
    return {
      title: '数据指标标题',
      tableTitle: '数据指标表格标题',
      chartTitle: '数据指标趋势图标题',
      columns: [
        {
          label: '日期',
          prop: 'date',
        },
        {
          label: '总数(台)',
          prop: 'total',
        },
        {
          label: '故障数量(台)',
          prop: 'failure',
        },
        {
          label: '占比(%)',
          prop: 'ratio',
        },
      ],
    };
  },
  methods: {
    // 覆写基类的事件处理方法,根据该统计指标请求不同的业务接口
    onFactoryChange() {},
    onChartTimeChange() {},
    onTableDateChange() {},
  },
};
</script>
<style lang="scss" scoped>
</style>

基类组件代码大致如下:

<!-- ReportBase.vue -->
<template>
  <div class="report-page router-page widget">
    <div class="widget-header">
      <div class="widget-header-title">{{title}}</div>
    </div>
    <div class="widget-body">
      <div class="filter flexbox">
        <span class="filter-name">{{ $t('pool.farm') }}</span>
        <el-radio-group class="plain" size="mini" v-model="factoryId" @change="onFactoryChange">
          <el-radio-button
            v-for="(item, index) in factoryList"
            :label="item.id"
            :key="index"
          >{{ item.name }}</el-radio-button>
        </el-radio-group>
      </div>
      <section class="chart-area section">
        <div class="flexbox align-center">
          <div class="chart-title">{{chartTitle}}</div>
          <el-radio-group size="mini" v-model="timespan" @change="onChartTimeChange">
            <el-radio-button label="week">周</el-radio-button>
            <el-radio-button label="month">月</el-radio-button>
            <el-radio-button label="year">年</el-radio-button>
          </el-radio-group>
        </div>
      </section>
      <section class="grid-area section">
        <div class="flexbox space-bt">
          <div class="grid-title">{{tableTitle}}</div>
          <el-date-picker
            v-model="month"
            type="month"
            placeholder="选择月"
            @change="onTableDateChange"
          ></el-date-picker>
        </div>
        <el-table :data="tableData" style="width: 100%">
          <el-table-column v-bind="item" v-for="(item, index) in columns" :key="index"></el-table-column>
        </el-table>
      </section>
    </div>
  </div>
</template>
<script>
import { mapGetters } from 'vuex';
export default {
  data() {
    return {
      title: '',
      factoryId: 0,
      month: '',
      tableTitle: '',
      columns: [],
      tableData: [],
      timespan: 'week',
    };
  },
  watch: {
    factoryList(value) {
      if (value.length > 0) {
        this.factoryId = value[0].id;
        this.onFactoryChange(this.factoryId);
      }
    },
  },
  computed: {
    ...mapGetters(['factoryList']),
  },
  methods: {
    onFactoryChange() {},
    onChartTimeChange() {},
    onTableDateChange() {},
  },
  created() {
    if (this.factoryList.length > 0) {
      this.factoryId = this.factoryList[0].id;
      this.onFactoryChange(this.factoryId);
    }
  },
};
</script>
<style lang="scss">

这样,后续如果有更多的统计指标页面,只需要继承和覆写方法就行,而基类几乎不用改动。当然,Vue组件的这种继承方式也不是完美的,比如 HTML模板要么完全继承,要么完全重写,不能按需继承某个部分。如果子类在结构上跟基类有所差异,还是需要在基类中做条件判断。如果模板差异太大,可以重新定义子类自己的template,至少还可以重用一部分业务逻辑代码。像本文提到的应用场景,就非常适合用继承来实现。

各位看官如果有更好的思路,欢迎在评论中一起探讨!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 162,475评论 4 372
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,744评论 2 307
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 112,101评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,732评论 0 221
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 53,141评论 3 297
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 41,049评论 1 226
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,188评论 2 320
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,965评论 0 213
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,716评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,867评论 2 254
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,341评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,663评论 3 263
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,376评论 3 244
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,200评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,990评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,179评论 2 285
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,979评论 2 279

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,041评论 1 32
  • CSS3是CSS技术的升级版本,CSS3语言开发是朝着模块化发展的。以前的规范作为一个模块实在是太庞大而且比较复杂...
    Naeco阅读 225评论 0 2
  • 2009年我步入了梦寐以求的大学,终于走出了小镇子来到了大城市打拼!那一天我就想未来的我一定属于这座城市........
    花残泪殇阅读 213评论 0 0
  • 自认为文章写作能力不好,以前每天都会写写日记什么的,后来也不怎么写,一到想写东西,不是这里卡壳了,就是那里死...
    甘恩阅读 83评论 0 0
  • 跑步圣经 在希腊奥林匹亚阿尔菲斯河岸的岩壁上,至今还刻着古希腊的一段格言:“如果你想聪明,跑步吧;如果你想强壮,跑...
    一枚冰儿阅读 539评论 0 3