发布订阅模式在vue中的实际运用实例详解


Posted in Javascript onJune 09, 2019

订阅发布模式定义了一种一对多的依赖关系,让多个订阅者对象同时监听某一个主题对象。这个主题对象在自身状态变化时,会通知所有订阅者对象,使它们能够自动更新自己的状态。

比如addEventListener 这个api就是个发布订阅模式

如果用过vue的同学,可以把他类比于 watch

下面我们看一个例子

var observe={
  fnsObj:{},
  // 订阅方法
  on:function(key,fn){
    if(!observe.fnsObj[key]){
      observe.fnsObj[key] = []
    }
    observe.fnsObj[key].push(fn)
  },
  // 发布
  emmit:function(key,value){
    if(observe.fnsObj[key].length){
      var fnsList = observe.fnsObj[key]
      for(var i=0;i<fnsList.length;i++){
        fnsList[i].call(this,value)
      }
    }
    
  },
  //删除订阅者
  remove:function(key){
    for(var k in observe.fnsObj){
      if(k===key) delete observe.fnsObj[key]
    }
  }
}

ok,那我们来尝试来调用下

// 我们订阅了两个监听者
observe.on('say',function(e){
console.log('i can hear he say: '+e)
})
observe.on('say',function(e){
console.log('he say: '+e)
})

接着发布消息

// 发布消息
observe.emmit("say","嘿,这里是发布者")

可以看见控制台返回了两条消息,就是刚才我们定义的订阅者里打印出的内容

i can hear he say: 这里是发布者

 he say: 这里是发布者

这就是发布订阅模式,我相信很多人概念都知道,但是至于在项目中如何实际运用,这又是个大问题。

毕竟设计模式感觉不是很常用,而且即使不用设计模式,也能实现需求,所以下面我着重介绍下,我在vue中碰到的一个需求中是如何使用发布订阅模式。

实际运用

1,需求介绍

我这个项目是公司内部的人力资源管理系统。因此需要根据对不同权限进行菜单获取,还有一些下拉框数据,以及一些基础信息,需要在登陆后就马上获取,在调用接口后获取数据后,把它存储在vuex里面

目前这几个方法是写在app.vue里面

// 获取基本数据
 this.loadMenu()
 this.loadBasicData()
 this.loadUserInfo()

所以我要考虑到两种情况

只有登陆后才能拉取这些数据

当前页面刷新,如果为登陆后则需要重新拉取数据,否则不进行任何操作

1,常规解决方案

这个算是比较普遍的需求,类似的很常见,按照常规解决方法,可以这么做:

在mixin 里面把这些方法放在里面

登陆成功后存一个状态到sessionStorage里面,同时调用这些方法拉取数据
app.vue里面的created生命周期里判断sessionStorage里的状态是否为登陆,如登陆则拉取数据
ok,需求解决,但是问题是,万一这些方法是只能放在app.vue里面呢,比如这个项目,app.vue不是我写的。是另外一个前端写,他不愿意把这些方法放在mixin呢?

如果我们能够监听sessionStorage的变化就可以了。但是无论是watch 还是computed 都没办法监听sessionStorage的变化,

所以这时候我们可以尝试使用 发布订阅模式

1,创建一个observe.js

class Observe {
 constructor() {
  this.fnsObj = {}
 }
   // 订阅方法
 on(key, fn) {
  if (!this.fnsObj[key]) { this.fnsObj[key] = [] }
  this.fnsObj[key].push(fn)
 }
   // 发布
 emmit(key, value) {
  if (this.fnsObj[key].length) {
   var fns = this.fnsObj[key]
   for (let i = 0; i < fns.length; i++) {
    fns[i].call(this, value)
   }
  }
 }
   //删除订阅者
 remove(key) {
  for (var k in this.fnsObj) {
   if (k === key) {
    delete (this.fnsObj[k])
   }
  }
 }
}
const observeSession = new Observe()

export default observeSession

2,在app.vue将他引入,同时定义监听和发布者

import observeSession from './utils/Observe'
...
created(){
  //刷新后如果为登陆状态则获取数据
   sessionStorage.getItem('login') === 'login' && this.loadSelectVal()
  //定义全局方法,在调用window.setSessionStorage的时候,发布者发布信息
   window.setSessionStorage = (key, value) => {
    observeSession.emmit('watchSesStore', { key, value })
   }
 // 监听存储在sessionStorage登陆状态变化,如果为登陆状态则获取数据,监听者监听信息
   observeSession.on('watchSesStore', e => {
    sessionStorage.setItem(e.key, e.value)
    e.value === 'login'&&this.loadSelectVal()
   })
}

最后我们在登陆login.vue页面登陆成功的时候,给他加一行

window.setSessionStorage('login', 'login')

以及在router.js路由控制里面,对退出的时候处理

//如果跳转到登陆页面则登陆状态为登出
 if (to.name === "Login") {
  next()
  window.setSessionStorage && window.setSessionStorage('login', 'logout')
 } else {
 ....

至此需求完美解决。

总结

以上所述是小编给大家介绍的发布订阅模式在vue中的实际运用实例详解,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

Javascript 相关文章推荐
ie 处理 gif动画 的onload 事件的一个 bug
Apr 12 Javascript
JQuery扩展插件Validate—4设置错误提示的样式
Sep 05 Javascript
简单谈谈jQuery(function(){})与(function(){})(jQuery)
Dec 19 Javascript
JS使用oumousemove和oumouseout动态改变图片显示的方法
Mar 31 Javascript
JS常用加密编码与算法实例总结
Dec 22 Javascript
移动端触摸滑动插件swiper使用方法详解
Aug 11 Javascript
inner join 内联与left join 左联的实例代码
Sep 18 Javascript
详解使用React进行组件库开发
Feb 06 Javascript
详解微信小程序之scroll-view的flex布局问题
Jan 16 Javascript
微信小程序实现提交input信息到后台的方法示例
Jan 19 Javascript
javascript实现页面的实时时钟显示示例
Aug 06 Javascript
vue-cil之axios的二次封装与proxy反向代理使用说明
Apr 07 Vue.js
为vue项目自动设置请求状态的配置方法
Jun 09 #Javascript
解决vue单页面应用中动态修改title问题
Jun 09 #Javascript
Vue动态修改网页标题的方法及遇到问题
Jun 09 #Javascript
详解Node.js异步处理的各种写法
Jun 09 #Javascript
在mpvue框架中使用Vant WeappUI组件库的注意事项【推进】
Jun 09 #Javascript
微信小程序发布新版本时自动提示用户更新的方法
Jun 07 #Javascript
微信小程序与webview交互实现支付功能
Jun 07 #Javascript
You might like
用DBSQL类加快开发MySQL数据库程序的速度
2006/10/09 PHP
匹配csdn用户数据库与官方用户的重合度并将重叠部分的用户筛选出来
2011/12/25 PHP
PHP设计模式之适配器模式代码实例
2015/05/11 PHP
PHP常见的几种攻击方式实例小结
2019/04/29 PHP
PHP数组基本用法与知识点总结
2020/06/02 PHP
Javascript 实现TreeView CheckBox全选效果
2010/01/11 Javascript
js分解url参数(面向对象-极简主义法应用)
2012/08/09 Javascript
解析jquery获取父窗口的元素
2013/06/26 Javascript
分享五个有用的jquery小技巧
2015/10/08 Javascript
最全的Javascript编码规范(推荐)
2016/06/22 Javascript
Ajax+FormData+javascript实现无刷新表单信息提交
2016/10/24 Javascript
微信小程序 Windows2008 R2服务器配置TLS1.2方法
2016/12/05 Javascript
详解ES6中的let命令
2020/04/05 Javascript
详解创建自定义的Angular Schematics
2018/06/06 Javascript
使用json-server简单完成CRUD模拟后台数据的方法
2018/07/12 Javascript
详解关于element el-button使用$attrs的一个注意要点
2018/11/09 Javascript
VUE 单页面使用 echart 窗口变化时的用法
2020/07/30 Javascript
Python内置函数——__import__ 的使用方法
2017/11/24 Python
Python线上环境使用日志的及配置文件
2019/07/28 Python
python 并发编程 多路复用IO模型详解
2019/08/20 Python
Python3读写Excel文件(使用xlrd,xlsxwriter,openpyxl3种方式读写实例与优劣)
2020/02/13 Python
如何基于python实现不邻接植花
2020/05/01 Python
Python3开发环境搭建详细教程
2020/06/18 Python
解决django migrate报错ORA-02000: missing ALWAYS keyword
2020/07/02 Python
Python基于xlutils修改表格内容过程解析
2020/07/28 Python
详解css3 object-fit属性
2018/07/27 HTML / CSS
Sephora丝芙兰印尼官方网站:购买化妆品和护肤品
2018/07/02 全球购物
zooplus意大利:在线宠物商店
2019/08/07 全球购物
美国农场商店:Blain’s Farm & Fleet
2020/01/17 全球购物
升职自荐书范文
2013/11/28 职场文书
小学音乐教学反思
2014/02/05 职场文书
高中军训感想800字
2014/02/23 职场文书
食品业务员岗位职责
2014/03/18 职场文书
乒乓球兴趣小组活动总结
2014/07/08 职场文书
解决numpy和torch数据类型转化的问题
2021/05/23 Python
Redis 常见使用场景
2021/08/30 Redis