vue封装一个通过API形式调用的组件 (单例模式思路)

文件结构
├── src
    └── assets  项目静态资源
        ├── logo.png  图片资源
    ├── components  业务组件
        ├── file  组件
          ├── file.js 
          ├── file.vue 
          └── index.js
    ├── view  页面文件
        └── home.vue
$file实现步骤
  • 声明一个实例 Instance
  • 然后将声明的实例挂载函数$mount()上
  • 获取Vue实例关联的DOM元素 然后将获取的DOM结构挂载到body上
file.vue
<template>
  <div class="preview" v-if="visible">
    <div class="preview_mask" @click="handleClose" />
    <div class="content">
      <img :src="src" alt />
    </div>
  </div>
</template>
<script>
export default {
  data() {
    return {
      src: '',
      visible:false,
    }
  },
  props: {
    fileType: {
      type: String,
      default: 'img',
    },
    fileSrc: {
      type: String,
      default: '',
    },
  },
  mounted() {
    this.showView()
  },
  methods: {
    handleClose() {
      this.visible = false
    },
    showView() {
      if (!this.fileSrc) {
        const message =
          this.fileType === 'img' ? '暂无相关影像信息' : '暂无相关文件'
        alert(message)
        return
      }
      if (this.fileType === 'img') {
        this.src = this.fileSrc
         this.visible = true
      } else {
        window.open(this.fileSrc)
      }
    },
  },
  // computed: {
  //   list() {
  //     return this.url ? [this.url] : []
  //   },
  // },
}
</script>
<style lang="less" scoped>
.preview {
  position: fixed;
  top: 0;
  right: 0;
  bottom: 0;
  left: 0;
  z-index: 9999;
  .preview_mask {
    position: absolute;
    top: 0;
    left: 0;
    right: 0;
    bottom: 0;
    background: rgba(55, 55, 55, 0.6);
    cursor: pointer;
  }
  .content {
    max-width: 90%;
    position: absolute;
    left: 50%;
    top: 50%;
    transform: translate3d(-50%, -50%, 0);
    img {
      max-width: 1000px;
    }
  }
}
</style>
file.js
import File from './file.vue'
import Vue from 'vue'

File.newInstance = properties => {
  const props = properties || {}
  const Instance = new Vue({
    data: props,
    render(h) {
      return h(File,{ props })
    }
  })

  const component = Instance.$mount()
  document.body.appendChild(component.$el)
}
export default File
index.js 封装api
import File from './file.js'

let fileInstance

function getFileInstance (props) {
  fileInstance = fileInstance || File.newInstance(props)
  return fileInstance
}


function file(props) {
  getFileInstance(props)
}

export default file
main.js
import Vue from 'vue'
import App from './App.vue'
// import Viewer from 'v-viewer'
import File from '@/components/file'  // 引入组件file
Vue.config.productionTip = false

// Vue.use(Viewer)

Vue.prototype.$File = File // 挂载到原型
new Vue({
  render: h => h(App),
}).$mount('#app')
index.vue 中使用 直接调用this.$File()
<template>
  <div>
    <button @click="handleOpen">打开文件</button>
  </div>
</template>

<script>
export default {
  name: 'demo',
  data() {
    return {
      fileSrc: require('./assets/logo.png')
    }
  },
  methods: {
    handleOpen() {
       this.$File({ fileSrc: this.fileSrc, fileType: 'img'})
    },
  }
}
</script>
使用单例模式实现项目中通用的唯一组件,这种方式封装成插件...开箱即用,可以根据自己的业务需求扩展

文字不够解释一下单例模式(这个是看的别的文字总结的)

单例模式

单例模式(Singleton),也叫单子模式,是一种常用的软件设计模式。在应用这个模式时,单例对象的类必须保证只有一个实例存在。许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息。这种方式简化了在复杂环境下的配置管理

优点
  • 在单例模式中,活动的单例只有一个实例,对单例类的所有实例化得到的都是相同的一个实例。这样就 防止其它对象对自己的实例化,确保所有的对象都访问一个实例
  • 单例模式具有一定的伸缩性,类自己来控制实例化进程,类就在改变实例化进程上有相应的伸缩性
  • 由于在系统内存中只存在一个对象,因此可以 节约系统资源,当 需要频繁创建和销毁的对象时单例模式无疑可以提高系统的性能
  • 避免对共享资源的多重占用
缺点
  • 不适用于变化的对象,如果同一类型的对象总是要在不同的用例场景发生变化,单例就会引起数据的错误,不能保存彼此的状态
  • 由于单利模式中没有抽象层,因此单例类的扩展有很大的困难
  • 单例类的职责过重,在一定程度上违背了“单一职责原则”
  • 滥用单例将带来一些负面问题,如为了节省资源将数据库连接池对象设计为的单例类,可能会导致共享连接池对象的程序过多而出现连接池溢出;如果实例化的对象长时间不被利用,系统会认为是垃圾而被回收,这将导致对象状态的丢失
适用场景

单例模式只允许创建一个对象,因此节省内存,加快对象访问速度,因此对象需要被公用的场合适合使用,如多个模块使用同一个数据源连接对象等等
1.需要频繁实例化然后销毁的对象。
2.创建对象时耗时过多或者耗资源过多,但又经常用到的对象。
3.有状态的工具类对象。
4.频繁访问数据库或文件的对象

posted @ 2021-04-21 22:32  _skylar  阅读(2358)  评论(0编辑  收藏  举报