DOM – MutationObserver
介绍
它和 IntersectionObserver, ResizeObserver 差不多, 都是观察 element 变化的.
它可以观察元素的 attribute 增加, 移除, 修改, append child 等等.
建议先看前 2 篇 IntersectionObserver 和 ResizeObserver 一起了解会比较容易.
new MutationObserver()
const mo = new MutationObserver((mutations) => {
console.log("mutations", mutations);
});
mo.observe(document.querySelector(".container"), {
attributes: true,
});
调用方式和 Intersection, ResizeObserver 是一样的.
注意: 它和 IO 和 Resize 有个区别, IO, Resize 调用 observe 第一次会马上触发掉, 但 Mutation 没有, 它会等到真的有改变时才触发.
还有一个特别之处是 observe 的时候需要一个 config. 指定要观察的范围.
出于性能考虑, 观察的范围越广性能越伤, 所以请按需设置哦.
观察 attributes
attributes: true
可以观察到元素添加, 移除, 修改 attribute. (注: 只要有 value set 即便 value 是一样的, 它依然会触发哦, 如果不想这样, 我们可以通过 oldValue 来 filter 掉)

attributeOldValue: true

多了一个 oldValue
attributeFilter: ["contenteditable"]
指定要观察的 attribute, 没有在 list 里面的, 添加, 移除, 修改都不会触发.
观察 Child 和 Descendant Element
childList: true
当元素 appendChild / removeChild 的时候触发.
subtree: true
当元素有子孙后裔插入或移除时触发.

观察 TextNode textcontent
characterData: true & characterDataOldValue: true
const textNode = document.createTextNode("Hello World");
mo.observe(textNode, { characterData: true, characterDataOldValue: true });
setTimeout(() => {
textNode.textContent = "SuperMan";
}, 3000);
效果

注意, 一定要是 TextNode 哦.
const p = document.createElement("p");
p.textContent = "Hello World";
mo.observe(p, { characterData: true, characterDataOldValue: true });
setTimeout(() => {
p.textContent = "SuperMan";
document.body.append(p);
}, 3000);
换成 p 就不灵了.

虽然 p 是有效的, 可以 append to body 看到字, 但是 observer 没有监听到它. 所以监听的节点一定要是 TextNode.
触发时机
和 IntersectionObserver,ResizeObserver 不同。
MutationObserver 触发的很早,它类似 Microtask。
而 IntersectionObserver,ResizeObserver 则是 after ui render。
也合理啦,毕竟 MutationObserver callback 获取的资料比如 addedNodes, removedNode 这些都不需要等 ui render。
window.setTimeout(() => {
const container = document.querySelector<HTMLElement>('.container')!;
const mo = new MutationObserver(
records => console.log(records[0].addedNodes.length > 0 ? 'mutation add' : 'mutation remove'), // 2, 4
);
mo.observe(container, { childList: true, subtree: true });
requestAnimationFrame(() => console.log('rAF')); // 6
const h1 = container.querySelector('h1')!;
h1.remove();
console.log('sync'); // 1
queueMicrotask(() => {
console.log('micro1'); // 3
container.appendChild(h1);
queueMicrotask(() => {
console.log('micro2'); // 5
});
});
}, 2000);
效果

相当于,h1.remove 之后它就 queueMicrotask for mutation callback。
没有 unobserve 方法,但有 takeRecords 方法
相关 Issue – Disconnect single target instead of all
不像 IntersectionObserver 和 ResizeObserver 有 unobserve 可以取消指定 element 的监听,
MutationObserver 只有 disconnect 一次性取消所有监听。
workaround 的方案是 wrap 一层,然后自己记入 element,unobserve 时,调用 disconnect 然后在 re-observe 其它的回去。参考高赞回复。
另外 MutationObserver 有一个 takeRecords 方法,这个方法用在 disconnect 之前,因为 MutationObserver 触发是 microtask,
所以在同步执行过程中,如果你 disconnect,这时是有可能已经有一些 MutationRecord 即将要发布的,而这个 takeRecords 方法就可以把它们拿出来处理,然后你才 disconnect。
DOM – MutationObserver的更多相关文章
- HTML5新特性之Mutation Observer
Mutation Observer(变动观察器)是监视DOM变动的接口.当DOM对象树发生任何变动时,Mutation Observer会得到通知. 要概念上,它很接近事件.可以理解为,当DOM发生变 ...
- livequery源码解读
从使用说起: 若干年前,有一天发现,通过js代码创建的html元素及ajax加载的html,无法被$([selector]).click(function(){...})绑定上事件,于是发现了jQue ...
- js实现数据视图双向绑定原理
这个方法了不起啊..vue.js和avalon.js 都是通过它实现双向绑定的..而且Object.observe也被草案发起人撤回了..所以defineProperty更有必要了解一下了几行代码看他 ...
- codepen & js demos
codepen & js demos Mutation Observer & customize resize event listener & demo https://co ...
- window resize & resize observer
window resize & resize observer https://developer.mozilla.org/en-US/docs/Web/API/Window/resize_e ...
- 强大的DOM变化观察者MutationObserver
在这之前 DOM3 提供了 Mutation events 事件 DOMAttrModified DOMAttributeNameChanged DOMCharacterDataModified DO ...
- 使用MutationObserver对象封装一个监听DOM生成的函数
(function(win){ 'use strict'; var listeners = []; var doc = win.document; var MutationObserver = win ...
- JavaScript是如何工作的:使用MutationObserver跟踪DOM的变化
摘要: 掌握MutationObserver. 这是专门探索 JavaScript 及其所构建的组件的系列文章的第10篇. 如果你错过了前面的章节,可以在这里找到它们: JavaScript 是如何工 ...
- js dom 观察者属性 MutationObserver
MDN上说的很清楚 MutationObserver给开发者们提供了一种能在某个范围内的DOM树发生变化时作出适当反应的能力.该API设计用来替换掉在DOM3事件规范中引入的Mutation事件 co ...
- MutationObserver DOM变化的观察
简单的给MutationObserver做个测试及总结笔记. MutationObserver,window上的一个(构造)函数,可以通过其创建的观察者(观察对象)达到观察DOM的变化的效果. 可适用 ...
随机推荐
- 首届 DIVE 精彩回顾丨践行企业数字化,基础软件如何创新
"墙高基下,虽得必失."在构建数字企业大厦的工程中,基础软件的重要性不言而喻.但对于各行各业而言,面向传统经营模式设计的基础软件已经难以支撑数字业务的创新,唯有汲取专业团队的经验, ...
- 【漏洞分析】Li.Fi攻击事件分析:缺乏关键参数检查的钻石协议
背景信息 2024 年 7 月 16日,Li.Fi 协议遭受黑客攻击,漏洞成因是钻石协议中 diamond 合约新添加的 facet 合约没有对参数进行检查,导致 call 函数任意执行.且 diam ...
- 逆向动态加载Dex(内存加载class)
逆向一个app, 其核心算法是通过反射调用的, 反编译软件中无法找到该类, 并且也无法hook. Java.perform(function(){ Java.enumerateClassLoaders ...
- 【转载】 ReLu(Rectified Linear Units)激活函数
原文地址: https://www.cnblogs.com/neopenx/p/4453161.html ============================== 论文参考:Deep Sparse ...
- ubuntu18.04 源码方式安装wine , 警告,libxrender 64-bit development files not found, XRender won't be supported.
警告信息: configure: WARNING: libxrender 64-bit development files not found, XRender won't be supported. ...
- 再探 游戏 《 2048 》 —— AI方法—— 缘起、缘灭(1) —— Firefox浏览器下自动运行游戏篇
四年前曾经写过一过博客: 对 游戏 < 2048 > 的一些思考 虽然过去几年了,但是这个游戏一直没有搞懂该怎么使用AI算法来进行求解,这里再次对这个问题进行一些探索. ========= ...
- c#12 实验特性Interceptor如何使用的一个简单但完整的示例
一直有很多转载dotnet对Interceptor说明文档的,但鲜有说明Interceptor如何使用的,这里写一篇简单示例来展示一下 c# 12 实验特性Interceptor 是什么? 官方解释如 ...
- Apache DolphinScheduler 在奇富科技的首个调度异地部署实践
奇富科技(原360数科)是人工智能驱动的信贷科技服务平台,致力于凭借智能服务.AI研究及应用.安全科技,赋能金融机构提质增效,助推普惠金融高质量发展,让更多人享受到安全便捷的金融科技服务.作为国内领先 ...
- 代码随想录Day11
150. 逆波兰表达式求值 给你一个字符串数组 tokens ,表示一个根据 逆波兰表示法 表示的算术表达式. 请你计算该表达式.返回一个表示表达式值的整数. 注意: 有效的算符为 '+'.'-'.' ...
- FlashAttention简介
前置知识 在GPU进行矩阵运算的时候,内部的运算单元具有和CPU类似的存储金字塔. 如果采用经典的Attention的计算方式,需要保存中间变量S和注意力矩阵O,这样子会产生很大的现存占用,并且这些数 ...