当前位置: 首页 > news >正文

发布订阅模式

一、常见的发布订阅模式

1、Dom的事件

Event + addEventListener + dispatchEvent

//订阅中心
const event = new Event('zyk');
//订阅
document.addEventListener('zyk', (value)=>{console.log('我收到了:', value)
});
//发布
document.dispatchEvent(e, '1');
document.dispatchEvent(e, '2');

遍历小工具(免密登录):

        通过Dom的事件中心,自动填充用户名、密码,无需手动输入。把以下代码添加收藏到网页,每次点击页面,就可以触发

javascript:
var inputEvent = documnet.createEvent('Event');
inputEvent.initEvent('init');
//获取用户名、密码、登录按钮的Dom
var userInput = document.getElementsByName('usernameInput')[0];
var pwdInput = document.getElementsByName('passwordInput')[0];
var loginButton = document.getElementsByName('loginBtn')[0];
//设置用户、密码的值
userInput.value = 'zyk';
pwdInput.value = '123456';
//发布用户名、密码的事件
userInput.dispatchEvent(inputEvent);
pwdInput.dispatchEvent(inputEvent);
//触发登录按钮的click事件
loginButton.click();

二、手写一个发布订阅中心

思路:

        模仿Event+addEventListener+dispatchEvent,我们的发布订阅库需要以下方法:

  • on = (event: string, callback: Function):void=>{}  //订阅
  • emit = (event: string, ...args): void =>{} //进行发布
  • off = (event: string, callback: Function): void => {} //取消订阅
  • once(event: string, callback: Function):void=>{}  //无论发布多少次,只订阅一次

实现代码:

class EventEmitter {constructor() {this.events = new Map();}on(event, callback) {if (this.events.has(event)) {let cbList = this.events.get(event);cbList.push(callback);this.events.set(event, cbList);} else {this.events.set(event, [callback]);}}emit(event, ...args) {let cbList = this.events.get(event);if (Array.isArray(cbList) && cbList.length > 0 ) {cbList.forEach(cb => cb(...args));}}off(event, callback) {let cbList = this.events.get(event);if(Array.isArray(cbList) && cbList.length >0) {cbList = cbList.filter(cb => cb!==callback);this.events.set(event, cbList);}}once(event, callback) {const fn = (...args) => {callback(...args); //执行一次this.off(event, fn);}this.on(event, fn)}
}

使用:

【1】发布多次,都可以一个他订阅

【2】发布1次,可以被多个订阅到

【3】取消一个订阅

【4】只订阅一次

http://www.lryc.cn/news/348749.html

相关文章:

  • 【Java 演示灵活导出数据】
  • 一对一WebRTC视频通话系列(六)——部署到公网
  • 【数据结构】线性表----链表详解
  • 【小程序】小程序如何适配手机屏幕
  • 第15节 编写shellcode加载器
  • JAVA学习-练习试用Java实现爬楼梯
  • [SWPUCTF 2021 新生赛]PseudoProtocols、[SWPUCTF 2022 新生赛]ez_ez_php
  • Hive-拉链表的设计与实现
  • AI应用案例:会议纪要自动生成
  • 基于光纤技术的新能源汽车电池安全监测--FOM²系统
  • 基于 LlaMA 3 + LangGraph 在windows本地部署大模型 (二)
  • Spring MVC(三) 参数传递
  • 常见加解密算法02 - RC4算法分析
  • 如何使用 ERNIE 千帆大模型基于 Flask 搭建智能英语能力评测对话网页机器人(详细教程)
  • Java全局异常处理,@ControllerAdvice异常拦截原理解析【简单易懂】
  • 代码随想录35期Day38-Java(Day37休息)
  • 力扣HOT100 - 739. 每日温度
  • 【爬虫之scrapy框架——尚硅谷(学习笔记one)--基本步骤和原理+爬取当当网(基本步骤)】
  • C++ QT设计模式:责任链模式
  • 基于springboot+mybatis+vue的项目实战之(后端+前后端联调)
  • 【教程向】从零开始创建浏览器插件(六)实战篇
  • 如何用 OceanBase做业务开发——【DBA从入门到实践】第六期
  • Element-UI快速入门
  • 【JavaWeb】网上蛋糕商城后台-商品管理
  • Django Admin后台管理:高效开发与实践
  • Centos7网卡启动失败(Failed to start LSB: Bring up/down)
  • 【NOIP2008普及组复赛】 题4:立体图
  • 【Leetcode每日一题】 动态规划 - 简单多状态 dp 问题 - 删除并获得点数(难度⭐⭐)(76)
  • Windows---CMD常用指令大全
  • 消息中间件是什么?有什么用?常见的消息中间件有哪些?