手写一个通用事件监听函数

在准备js面试题时,遇到了许多知识盲区,或是已经遗忘的知识,所以来写一下博客,记录自己的成长,同时查漏补缺

这次来写一个通用的事件监听函数,基于我们的事件冒泡,如果你不知道事件冒泡

需求,点击每个a,返回对应的内容

 <div id="div">
    <a id="p1" href="#">激活</a>
    <a id="p2" href="#">取消</a>
    <a id="p3" href="#">取消</a>
    <a id="p4" href="#">取消</a>
    <a id="p5" href="#">激活</a>
    <a id="p6" href="#">取消</a>
    <a id="p7" href="#">取消</a>
    <a id="p8" href="#">取消</a>
    <button class="btn">加载更多...</button>
  </div>

先封装一个函数,看代码

function bindEvent(elem, type, fn) {
  elem.addEventListener(type,fn)
}
//调用时,如
bindEvent(div,'click',function(){})

是这样吗?当然不是,要是这么简单那面试时就没什么可问的了

我们写的要考虑上事件代理这个情况让这个函数可以使用事件代理,也可以不用事件代理

function bindEvent(elem, type, selector, fn) {
  //判断一下,如果我们只传入三个值,那就是普通函数
  //如果是传入四个参数,那就是使用事件代理
  //selector是我们要代理的对象,如果没有,那就不是使用事件代理
  if(fn == null) {
    fn = selector
    selector = null
  }
  elem.addEventListener(type, function(e) {
    const target = e.target
    if(selector) {
      //使用事件代理
      if(target.matches(selector)) {
        //这matches主要是用来判断当前DOM节点是否能完全匹配对应的CSS选择器规则;
         //如果匹配成功,返 回true,反之则返回false
        //可以理解为当我们传入的selector===target.nodeName,返回true,否则false
        fn.call(target, e)
        //使用call函数,方便我们使用this指向我们点击的元素
      }
    }else {
      //普通调用
      fn.call(target, e)
    }
  })
}

 使用代理时

bindEvent(div1,'click','a',function() {
  alert(this.innerHTML)
})
//点击每个a时,都可以alert出a的值

使用普通函数时

bindEvent(div1,'click',function(e) {
  const target = e.target
  if(target.nodeName === 'A') {
    alert(target.innerHTML)
  }
})
//点击每个a时,都可以弹出相对应的值

  • 6
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值