javascript中如何监听页面DOM变动并高效响应

技术javascript中如何监听页面DOM变动并高效响应本篇内容介绍了“javascript中如何监听页面DOM变动并高效响应”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一

本文介绍了“如何在javascript中监控页面DOM变化并有效响应”的知识。很多人在实际的案件操作过程中都会遇到这样的困难。接下来,让边肖带领大家学习如何应对这些情况!希望大家认真阅读,学点东西!

从 DOM 变动事件监听说起

首先假设大家都已经知道了JavaScript中事件的阶段(capture-* * *-冒泡),附上图片拍下这个内容,我们直接进入寻找解决方案的过程。

javascript中如何监听页面DOM变动并高效响应

使用DOM事件流在DOM树中调度的事件的图形表示

起初,我一直在搜索窗口状态变化所涉及的事件。经过一轮搜索,我发现onload事件是最接近的,那么我们来看看MDN对这个事件的定义:

当资源及其相关资源完成加载时,将触发加载事件。

如何理解资源及其依赖的资源已经被加载?简单来说,如果一个页面涉及图片资源,那么onload事件会在页面完全加载后触发(包括图片、css文件等)。).一个简单的监听事件应该用JavaScript编写如下(注意不同环境中加载和onload的区别):

script window . addeventlistener(' load '),function(event){ console . log(' allresourcesfinisheddloading!');});//or indow . onload=function(){ console . log(' Allresourcesfinisheddloading!');};//html body onload=' some JavaScript code '//jQuery $(window)。On ('load ',handler)/script当然,说到onload事件,jquery中有一个类似的事件肯定会提到mdashmdash就绪事件。这个事件在jQuery中的定义如下:

指定DOM完全加载时要执行的函数。

需要知道的是,jQuery定义的ready Event本质上是为DOMContentLoaded事件设计的,所以当我们谈到加载时,应该区分的事件实际上是onload(接口UIEvent)和DOMContentLoaded(接口事件)。MDN这样描述加载的多内容:

当初始HTML文档被完全加载和解析后,就触发了DOMContentLoaded事件,而无需等待样式表、图像和子帧完成加载。不同的事件load应该仅用于检测完全加载的页面。

因此,可以知道当加载一个页面时,应该在onload之前触发DOMContentLoaded。类似事件和差异包括以下类别:

DOMContentLoaded:当初始HTML文档完全加载并解析后,触发DOMContentLoaded事件,无需等待样式表、图像和子帧完成加载。

Readystatechange:文档的Document.readyState属性描述了

文档的加载状态,当这个状态发生了变化,就会触发该事件;

  • load: 当一个资源及其依赖资源已完成加载时,将触发load事件;

  • beforeunload: 当浏览器窗口,文档或其资源将要卸载时,会触发beforeunload事件。

  • unload: 当文档或一个子资源正在被卸载时, 触发 unload事件。

  • 细心点会发现上面在介绍事件时提到了 UIEvent 以及 Event,这是什么呢?这些都是事件——可以被 JavaScript  侦测到的行为。其他的事件接口还包括 KeyboardEvent / VRDisplayEvent (是的,没错,这就是你感兴趣且熟知的那个  VR)等等;如果在搜索引擎中稍加搜索,你会发现有些资料里写到事件可以分为以下几类:

    • UI事件

    • 焦点事件

    • 鼠标与滚轮事件

    • 键盘与文本事件

    • 复合事件

    • 变动事件

    • HTML5 事件

    • 设备事件

    • 触摸与手势事件

    但这样写实在有些凌乱,其中一些是 DOM3 定义的事件,有一些是单独列出的事件,如果你觉得熟悉那么你会发现这是 JavaScript  高级程序设计里的叙述模式,在我看来,理解这些事件可以按照 DOM3 事件以及其他事件来做区分:其中,DOM3 级事件规定了以下几类事件 – UI 事件,  焦点事件, 鼠标事件, 滚轮事件, 文本事件, 键盘事件, 合成事件, 变动事件, 变动名称事件; 而剩下的例如 HTML5 事件可以单独做了解。而刚开始提到的  Event 作为一个主要接口,是很多事件的实现父类。有关 Web API 接口可以在这里查到,里面可以看到有很多 Event 字眼。

    好吧,事件说了这么多,我们还是没有解决刚开始提出的问题,如果监听页面中动态生成的元素呢?想到动态生成的元素都是需要通过网络请求获取资源的,那么是否可以监听所有  HTTP 请求呢?查看 jQuery 文档可以知道每当一个Ajax请求完成,jQuery 就会触发 ajaxComplete  事件,在这个时间点所有处理函数会使用 .ajaxComplete() 方法注册并执行。但是谁能保证所有 ajax 都从 jQuery  走呢?所以应该在变动事件中做出选择,我们来看看 DOM2 定义的如下变动事件:

    • DOMSubtreeModified: 在DOM结构发生任何变化的时候。这个事件在其他事件触发后都会触发;

    • DOMNodeInserted: 当一个节点作为子节点被插入到另一个节点中时触发;

    • DOMNodeRemoved: 在节点从其父节点中移除时触发;

    • DOMNodeInsertedIntoDocument: 在一个节点被直接插入文档或通过子树间接插入文档之后触发。这个事件在  DOMNodeInserted 之后触发;

    • DOMNodeRemovedFromDocument: 在一个节点被直接从文档移除或通过子树间接从文档移除之前触发。这个事件在  DOMNodeRemoved 之后触发;

    • DOMAttrModified: 在特性被修改之后触发;

    • DOMCharacterDataModified: 在文本节点的值发生变化时触发;

    所以,用 DOMSubtreeModified 好像没错。师兄旁边提醒,用 MutationObserver, 于是又搜到了一个新大陆。MDN 这样描述  MutationObserver:

    MutationObserver给开发者们提供了一种能在某个范围内的DOM树发生变化时作出适当反应的能力.该API设计用来替换掉在DOM3事件规范中引入的Mutation事件.

    DOM3 事件规范中的 Mutation 事件可以被简单看成是 DOM2 事件规范中定义的 Mutation  事件的一个扩展,但是这些都不重要了,因为他们都要被 MutationObserver 替代了。好了,那么来详细介绍一下 MutationObserver  吧。文章《Mutation Observer API》对 MutationObserver  的用法介绍的比较详细,所以我挑几点能直接解决我们需求的说一说。

    既然要监听 DOM 的变化,我们来看看 Observer 的作用都有哪些:

    它等待所有脚本任务完成后,才会运行,即采用异步方式。

    它把 DOM 变动记录封装成一个数组进行处理,而不是一条条地个别处理 DOM 变动。

    它既可以观察发生在 DOM 的所有类型变动,也可以观察某一类变动。

    MutationObserver 的构造函数比较简单,传入一个回调函数即可(回调函数接受两个参数,***个是变动数组,第二个是观察器实例):

    let observer = new MutationObserver(callback);

    观察器实例使用 observe 方法来监听, disconnect 方法停止监听,takeRecords 方法来清除变动记录。

    let article = document.body;     let  options = {    'childList': true,    'attributes':true  } ;     observer.observe(article, options);

    observe 方法中***个参数是所要观察的变动 DOM 元素,第二个参数则接收所要观察的变动类型(子节点变动和属性变动)。变动类型包括以下几种:

    • childList:子节点的变动。

    • attributes:属性的变动。

    • characterData:节点内容或节点文本的变动。

    • subtree:所有后代节点的变动。

    想要观察哪一种变动类型,就在 option 对象中指定它的值为 true。需要注意的是,如果设置观察 subtree 的变动,必须同时指定  childList、attributes 和 characterData 中的一种或多种。disconnect 方法和 takeRecords  方法则直接调用即可,无传入参数。

    好的,我们已经搞定了 DOM 变动的监听,将代码刷新一下看下效果吧,因为页面由很多动态生成的商品组成,那么我应该在 body 上添加变动监听,所以  options 应该这样设置:

    var options = {    'attributes': true,    'subtree': true  }

    咦?页面往下拉一小点就触发了 observer 几十次?这样 DOM 哪吃得消啊,查看了页面的变动记录发现每次新进的资源底层都调用了  Node.insertBefore() 方法…

    再聊聊 JavaScript 中的截流/节流函数

    现在遇到的一个麻烦是, DOM 变动太频繁了,如果每次变动都监听那真是太耗费资源了。一个简单的解决办法是我就放弃监听了,而采用 setInterval  方法定时执行更新逻辑。是的,虽然方法原始了一点,但是性能上比 Observer “改进”了不少。

    这个时候,又来了师兄的助攻:“用用截流函数”。记起之前看《JavaScript 语言精粹》的时候看到是用 setTimeout 方法自调用来解决  setInteval 的频繁执行吃资源的现象,不知道两者是不是有关联。网上一查发现有两个“jie流函数”。需求来自于这里:

    在前端开发中,页面有时会绑定scroll或resize事件等频繁触发的事件,也就意味着在正常的操作之内,会多次调用绑定的程序,然而有些时候javascript需要处理的事情特别多,频繁出发就会导致性能下降、成页面卡顿甚至是浏览器奔溃。

    如果重复利用 setTimeout 和 clearTimeout 方法,我们好像可以解决这个频繁触发的执行。每次事件触发的时候我首先判断一下当前有没有一个  setTimeout 定时器,如果有的话我们先将它清除,然后再新建一个 setTimeout  定时器来延迟我的响应行为。这样听上去还不错,因为我们每次都不立即执行我们的响应,而频繁触发过程我们又能保持响应函数一直存在(且只存在一个),除了会有些延迟响应外,没什么不好的。是的这就是截流函数(debounce),有一篇博客用这个小故事介绍它:

    形像的比喻是橡皮球。如果手指按住橡皮球不放,它就一直受力,不能反弹起来,直到松手。debounce 的关注点是空闲的间隔时间。

    在我的业务中,在 observer 实例中调用下面写的这个截流函数就可以啦

    /**  * fn 执行函数  * context 绑定上下文  * timeout 延时数值  **/  let debounce = function(fn, context, timeout) {  let timer;            // 利用闭包将内容传递出去  return function() {    if (timer) {      // 清除定时器      clearTimeout(timer);    }    // 设置一个新的定时器    timer = setTimeout(function(){    fn.apply(context, arguments)    }, timeout);   }  }

    当然,解决了自己的问题,但还有一个概念没有说到——“节流函数”。同一篇博文里也使用了一个例子来说明它:

    形像的比喻是水龙头或机枪,你可以控制它的流量或频率。throttle 的关注点是连续的执行间隔时间。

    函数节流的原理也挺简单,一样还是定时器。当我触发一个时间时,先setTimout让这个事件延迟一会再执行,如果在这个时间间隔内又触发了事件,那我们就清除原来的定时器,再setTimeout一个新的定时器延迟一会执行。函数节流的出发点,就是让一个函数不要执行得太频繁,减少一些过快的调用来节流。这里用  AlloyTeam 的节流代码实现来解释:

    // 参数同上  var throttle = function(fn, delay, mustRunDelay){   var timer = null;   var t_start;   return function(){      var context = this, args = arguments, t_curr = +new Date();            // 清除定时器      clearTimeout(timer);            // 函数初始化判断      if(!t_start){          t_start = t_curr;      }            // 超时(指定的时间间隔)判断      if(t_curr - t_start >= mustRunDelay){          fn.apply(context, args);          t_start = t_curr;      }      else {          timer = setTimeout(function(){              fn.apply(context, args);          }, delay);      }   };  };

    “javascript中如何监听页面DOM变动并高效响应”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注网站,小编将为大家输出更多高质量的实用文章!

    内容来源网络,如有侵权,联系删除,本文地址:https://www.230890.com/zhan/93972.html

    (0)

    相关推荐

    • Python的缺点是什么

      技术Python的缺点是什么这篇文章主要介绍“Python的缺点是什么”,在日常操作中,相信很多人在Python的缺点是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python的缺点

      攻略 2021年10月25日
    • python多线程如何秒爬猫眼电影

      技术python多线程如何秒爬猫眼电影这篇文章主要介绍python多线程如何秒爬猫眼电影,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!通过python3的多线程,获取猫眼电影top100电影清单,

      攻略 2021年11月24日
    • 南昌旅游必去景点,南昌市内有那些旅游景点

      技术南昌旅游必去景点,南昌市内有那些旅游景点1南昌旅游必去景点、滕王阁 秋水共长天一色江南三大名楼之一,因初唐诗人王渤的《滕王阁序》而声名远播,历经29次兴废之后,她依然保留着古朴的风格,弥漫着一股人文之气。素有“西江第

      生活 2021年10月27日
    • MONGO DB too many mongodb page_faults 报错排查step by step的示例分析

      技术MONGO DB too many mongodb page_faults 报错排查step by step的示例分析这篇文章给大家介绍MONGO DB too many mongodb page_faults

      攻略 2021年11月3日
    • Phone类的怎么实现

      技术Phone类的怎么实现这篇文章主要讲解了“Phone类的怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Phone类的怎么实现”吧! Decorat

      攻略 2021年12月8日
    • android项目一般用到哪些开源库(android开源库分类及使用方法)

      技术实用的Android开源库有哪些这篇文章主要为大家展示了“实用的Android开源库有哪些”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“实用的Android开源库有哪些”

      攻略 2021年12月24日