我们知道,在JavaScript中,原生DOM事件在开发中是很有用的(与用户交互的重要方式),但是操作原生DOM事件其实有两大缺点:性能低、依赖于浏览器(NodeJs、小程序等不可用)。那么这个时候,就需要我们进行自定义事件去处理某些特定的业务。

认识Event对象及元素的dispatchEvent方法

在JavaScript中,所有事件的父对象就是Event对象,也就是说像我们平时所有的点击(click)、触摸(touch)、鼠标等事件对象都继承自Event。理解这一点是很重要的。先来简单看一个事件的场景。

场景一、页面上有两个按钮a、b,当点击按钮b的时候,调用按钮a的点击事件。简单布局代码如下:

<button id="btn1">a</button>
<button id="btn2">b</button>

程序员A的做法,分别获取这两个按钮,然后给b按钮添加点击事件后,调用按钮a的click方法。代码如下:

<button id="btn1" onclick="alert('I am a button named a')">a</button>
<button id="btn2">b</button>
<script>
let btn1 = document.querySelector('#btn1');
let btn2 = document.querySelector('#btn2');
btn2.onclick = function(){
btn1.onclick();
}
</script>

程序员B的做法,分别获取这两个按钮,然后给b按钮添加点击事件后,在回调函数中在添加按钮a的点击事件。代码如下:

<button id="btn1">a</button>
<button id="btn2">b</button>
<script>
let btn1 = document.querySelector('#btn1');
let btn2 = document.querySelector('#btn2');
btn2.onclick = function(){
btn1.addEventListener('click',function(){
alert('I am a button named a')
},false)
}
</script>

看到这里,你认为谁的做法是正确的?显然程序员A的做法是对的(就目前的要求来看),但有缺陷,如果按钮a的事件是通过addEventListener方法去注册监听的,就不起作用了。那么该怎样做才会比较好?这就需要我们的Event对象和元素的dispatchEvent方法了。改进代码如下:

<button id="btn1">a</button>
<button id="btn2">b</button>
<script>
let btn1 = document.querySelector('#btn1');
let btn2 = document.querySelector('#btn2');
btn1.addEventListener('click',function(){
alert('I am a button named a')
},false)
btn2.onclick = function(){
let ev = new Event('click');
btn1.dispatchEvent(ev);
}
</script>

其中:

  • Event对象的构造函数需要一个参数,事件类型
  • dispatchEvent方法是将某个事件分发给某个元素

认识自定义事件

前面说过,在浏览器端javascript中,所有的事件都继承自Event,那么其实要想实现一个自定义事件,也是需要继承自Event。

还是类似上面说过的场景,场景二:有两个按钮a,b,当调用b按钮的点击事件,怎么去触发a按钮上的自定义事件?

<button id="btn1">a</button>
<button id="btn2">b</button>
<script>
let a = document.querySelector('#btn1');
let b = document.querySelector('#btn2');
a.addEventListener('myClick',function(){
alert('I am a button named a')
},false)
class MyEvent extends Event{
constructor(...args){
super(...args);
this.a = 12;
}
}
b.onclick = function(){
const ev = new MyEvent('myClick');
a.dispatchEvent(ev);
}
</script>

这就是自定义事件的思想体现

  • 根据事件类型进行事件的注册
  • 根据事件的类型分发对应的事件给需要者

可以看出,对事件进行管理是很有必要,如Java中的EventBus、VueJs中的、emit等,将事件的监听者及分发者抽象成一个独立的模块,来进行事件的管理(添加、移除等)有利用解耦。

事件队列完成组件间的通信

这里可以把事件队列想象成一根管道,类似前端gulp实现的核心思想(基于管道)、当使用者需要使用某个事件的时候,就在管道中注册一个事件,然后通过该事件的类型,从管道中分发一个该类型的事件,在不需要使用后,还可以对相应的事件进行移除操作。代码如下:

class Pipe{
constructor(){
this.pipes = {};
}
/**
* 注册事件
* @param {*} type
* @param {*} fn
*/
on(type,fn){
this.pipes[type] = this.pipes[type] || [];
if(this.pipes[type].findIndex(func => func==fn)==-1){
this.pipes[type].push(fn);
}
}
/**
* 移除事件
* @param {*} type
* @param {*} fn
*/
off(type,fn){
if(this.pipes[type]){
this.pipes[type] = this.pipes[type].filter((func) => func!==fn); if(this.pipes[type].length===0){
delete this.pipes[type];
}
}
}
/**
* 分发事件
* @param {*} type
* @param {...any} args
*/
emit(type,...args){
if(this.pipes[type]){
this.pipes[type].forEach((fn) => {
fn(...args);
})
}
}
}

场景:通过事件队列模拟vuejs中组件间通信的实现。Component1负责对数据进行渲染,Component2中点击按钮,来改变Component1实例属性a的值。代码如下:

 
<div id="box">
{{a}}
</div>
<button id="btn1">++</button>
<script>
const $ = document.querySelectorAll.bind(document);
let pipe = new Pipe();
class Component1{
constructor(){
this.a = 12;
this.el = $("#box")[0];
this.render();
pipe.on('add',(arg) => {
this.a+=arg;
this.render();
})
} render(){
this.el.innerHTML = this.a;
}
} class Component2{
constructor(){
this.el = $("#btn1")[0];
this.el.onclick = function(){
pipe.emit('add',12);
}
}
} new Component1();
new Component2();
</script>

转发至:https://segmentfault.com/a/1190000020550517?utm_source=tag-newes

利用JavaScript自定义事件完成组件间的数据通信的更多相关文章

  1. Javascript事件模型系列(四)我所理解的javascript自定义事件

    被我拖延了将近一个月的javascript事件模型系列终于迎来了第四篇,也是我计划中的最后一篇,说来太惭愧了,本来计划一到两个星期写完的,谁知中间遇到了很多事情,公司的个人的,搞的自己心烦意乱浮躁了一 ...

  2. 理解的javascript自定义事件

    理解的javascript自定义事件 被我拖延了将近一个月的javascript事件模型系列终于迎来了第四篇,也是我计划中的最后一篇,说来太惭愧了,本来计划一到两个星期写完的,谁知中间遇到了很多事情, ...

  3. Javascript自定义事件功能与用法实例分析

    原文地址:https://www.jb51.net/article/127776.htm 本文实例讲述了javascript自定义事件功能与用法.分享给大家供大家参考,具体如下: 概述 自定义事件很难 ...

  4. javascript:自定义事件初探

    javascript:自定义事件初探   http://www.cnblogs.com/jeffwongishandsome/archive/2008/10/27/1317148.html

  5. javascript 自定义事件 发布-订阅 模式 Event

    * javascript自定义事件 var myEvent = document.createEvent("Event"); myEvent.initEvent("myE ...

  6. vuejs单一事件管理组件间的通信

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. 高级功能:很有用的javascript自定义事件

    之前写了篇文章<原生javascript实现类似jquery on方法的行为监听>比较浅显,能够简单的使用场景. 这里的自定义事件指的是区别javascript默认的与DOM交互的事件,比 ...

  8. JavaScript自定义事件,动态添加属性

    根据事件的不同,可用的自定义方法也不同. document.createEvent('Event'); 实现主要有4个步骤: 1.创建事件. 2.初始化事件(三个参数:事件名,是否起泡,是否取消默认触 ...

  9. vue自定义事件 子组件把数据传出去

    每个 Vue 实例都实现了事件接口(Events interface),即: 使用 $on(eventName) 监听事件 使用 $emit(eventName) 触发事件 1.使用v-on绑定自定义 ...

  10. JavaScript自定义事件

    很多DOM对象都有原生的事件支持,向div就有click.mouseover等事件,事件机制可以为类的设计带来很大的灵活性,相信.net程序员深有体会.随着web技术发展,使用JavaScript自定 ...

随机推荐

  1. 本地文件包含漏洞详解与CTF实战

    1. 本地文件包含简介 1.1 本地文件包含定义 本地文件包含是一种Web应用程序漏洞,攻击者通过操控文件路径参数,使得服务器端包含了非预期的文件,从而可能导致敏感信息泄露. 常见的攻击方式包括: 包 ...

  2. windows当中C++版本的Opencv安装(动态库+静态库)

    主要参考2篇博客,其实就是dll文件和lib文件的使用方法而已.链接如下: 1.静态库opencv配置 2.动态库opencv安装

  3. 研发LLM模型,如何用数值表示人类自然语言?

    上一篇:<人工智能--自然语言处理简介> 序言:人工智能大语言模型(LLM)如何理解人类的自然语言?这个过程的核心在于将文本转化为计算机能处理的数值形式,经过计算,最终达到对语言的理解.起 ...

  4. cornerstone中raft_server源码解析

    1.概述 cornerstone中核心即为raft_server的实现. 在raft里面有follower,leader,candidate三种角色,且角色身份还可以相互切换. 写三个类followe ...

  5. 教育账号无法登录OneDrive的一种解决方法

    众所周知,微软的服务总是能出现一些奇奇怪怪的问题,比如说教育账号无法登录OneDrive,尝试使用网上的临时解决方案失败 onedrive学生账号无法登录win10 OneDrive客户端 用户可以在 ...

  6. 静态分析工具及使用总结(二)CheckStyle

    这里主要介绍三种开源的工具,PMD.CheckStyle和FindBugs,着重是在Ant里的调用,据说商业软件JTest也是著名的代码分析工具,哈哈,要花钱的没有用过. Checkstyle (ht ...

  7. a标签与Blob下载文件的区别和获取文件下载进度

    文件下载的几种方式. 大家都做过文件下载,无非就是通过a标签给定一个href. 用户点击下载按钮. 或者使用Blob的方式进行下载. 这两种是很常见的,也是我们平时做使用最多的方式. 那么我们知道这2 ...

  8. Vue项目报TypeError: Cannot read properties of undefined (reading '_wrapper')

    前情 最近在做一个营销活动的时候,我选择了Vue技术栈来开发. 坑位 项目看似一切都正常,但当我在绑定的js事件中去修改当前组件的data上的值时会报错:TypeError: Cannot read ...

  9. Acrobat Pro DC 2024.005 像word一样编辑PDF

    随着数字化的推广,PDF文件凭借其强大的优势和稳定性逐渐成为各类文档交流和存储的首选格式.随之而来的是对PDF文件的阅读.编辑.转换.转曲等各种操作需求的不断增长.因此,一款强大的PDF处理软件不仅需 ...

  10. openEuler欧拉修改SSH端口

    修改SSH端口的主要原因是提高服务器的安全性.默认情况下,SSH服务运行在端口22上,因此攻击者和自动化脚本通常会针对此端口发起暴力破解攻击.密码猜测和其他恶意活动. vim /etc/ssh/ssh ...