zoukankan      html  css  js  c++  java
  • 手写事件代理函数 (Delegated function)

    ‘手写 ’ 这个词 ,面试是不是听过无数遍呢 !

    今天我们来手写一个这样的事件委托函数 =》

    function( parent, selector, type ,  handle)  {}

    你需要传递的参数分别是:

    parent: 事件绑定的父级

    selector: 选择器,

    type:  事件类型

    handle: 事件处理函数

    写之前 先了解一下事件委托的概念:     

        它是通过事件冒泡机制,

           即子元素上触发的事件会冒泡到父级上, 即父级也会触发该类型的事件,

           通过父级触发的事件拿到事件源对象e.target  就可以 知道 真正触发事件的元素是什么。

    举个例子, 一个ul 下面有 1000 个  li   , 我们 要给  li  绑定  点击事件 , 如果给每个li都绑定一个

    点击事件  会  大量占用 内存   , 不利于性能优化, 这种情况下 我们 只需要在ul上绑定一个点击事件,

    通过class 或者 id 来识别每个li ,  这样就大大减少了事件注册, 而且 再 有新增的li时 我们也无需再去注册点击事件

    我们来写个小demo

    HTML:

    <ul id="parent">
        <li>1</li>
        <li>2</li>
        <li>3</li>
    </ul>

    css:

    .active{
        background-color: green;
    }

    js:

    const parent = document.getElementById('parent');
    function changeColor() {
      if (this.classList.contains('active')) {
      this.classList.remove('active')
      } else {
      this.classList.add('active');
      }
     } 

    function delegate(parent, selector, type, handle) { }
    delegate(parent,
    'li', 'click',changeColor);

    我们要实现点哪个li  , 哪个li就变成原谅色, 再次点击取消。并 做 浏览器的兼容

    我们开始写 delegate函数:

    function matchSelector(target, selector) {
        if (selector.match(/^#/g)) { // 匹配以#开头的字符串
            return target.id === selector.slice(1); // 截取#后面的字符串
        }
        if (selector.match(/^./g)) {  // . 要转义,匹配以点开头的字符转
            return target.classList.contains(selector.slice(1))
        }
        return target.nodeName.toLowerCase() ===  selector; // 匹配标签
    }
    function delegate(parent, selector, type, handle) {
        if (parent.addEventListener) {
            parent.addEventListener(type, eventFn, false)
        } else {   // 兼容老IE
            parent.attachEvent('on' + type, handle)
        }
        function eventFn(e) {
            const event = e || window.event;
            const target = event.target || event.srcElement;// 兼容老IE
            if (matchSelector(target, selector)) {
                if (handle) {
                    handle.call(target, e); // 让handle执行时的this指向target
                }
            }
        }
    }

     效果:

               

  • 相关阅读:
    BZOJ 2789: [Poi2012]Letters 树状数组 + 逆序对
    luogu 5468 [NOI2019]回家路线 最短路/暴力
    BZOJ 2427: [HAOI2010]软件安装 tarjan + 树形背包
    练手爬虫用urllib模块获取
    django安装以及配置
    web框架
    模拟百度进行图片搜索,有问题可以留言
    深入requests库params|data|json参数
    深入理解http1.x、http 2和https
    nodejs 实现 磁力链接资源搜索 BT磁力链接爬虫
  • 原文地址:https://www.cnblogs.com/LHLVS/p/10712952.html
Copyright © 2011-2022 走看看