先查Network面板Initiator列定位触发DOM插入的JS代码,再重点检查innerHTML+=、重复调用注入函数、框架接管冲突及广告SDK重试未防重等问题。
直接看 Network 面板里的 Initiator 列——它会明确标出触发每个请求的 JS 调用栈。重点不是查“谁发了请求”,而是查“谁触发了 DOM 插入”,因为资源重试本身不直接导致节点重复,真正造成重复的是插入逻辑被多次执行。
Initiator 字段,点进去跳转到对应 JS 行eval 或压缩后的匿名函数,就往上翻调用栈,找最靠近顶层的、你可控的函数名(比如 injectAd、renderBanner)DOMContentLoaded 和 window.onload 里都调用了同一插入函数innerHTML += 看似简单,实则是 DOM 重复插入的高发操作。它每次执行都会先读取当前 innerHTML 字符串,再拼接新内容,最后整个替换——这会销毁原有节点、清空事件监听器,并让浏览器重新解析 HTML,极易引发副作用。
container.innerHTML += '<iframe src="ad.js"></iframe>':只要该语句被执行两次,就会插入两个 iframedocument.createElement('iframe') + appendChild(),或封装成“只插入一次”的幂等逻辑,例如加 data-injected="true" 标记并提前判断insertAdjacentHTML('beforeend', ...) 比 innerHTML += 稍安全,但依然不解决重复问题,只是避免重建整个容器结构如果你页面同时存在原生 JS 插入和前端框架接管的区域,框架的 diff 算法可能把原生插入的节点识别为“脏节点”而覆盖或复制——这不是 bug,而是 DOM 所有权冲突。
root 或 el 配置覆盖,比如 new Vue({ el: '#container' }) 会让 Vue 完全控制该节点及其子树patch、hydrate),说明是框架在重写 DOM;此时应改用框架提供的插槽(slot)或指令(如 v-html)注入广告,而非原生操作很多第三方广告 SDK 在加载失败后会自动重试,但它们的插入函数往往没做幂等保护。你不能改 SDK,但可以在调用层拦截。
立即学习“前端免费学习笔记(深入)”;
let injected = false; if (!injected) { injectAd(); injected = true; }
if (!document.querySelector('iframe[src*="ad-domain.com"]')) { injectAd(); }
id 判断——第三方常动态生成 ID,不可靠;优先用 src 特征、data-ad-type 属性或固定 class 名onload 回调里直接插入,而应在广告脚本 load 事件后、且确保 DOM 就绪后再执行插入真正麻烦的不是插入动作本身,而是插入时机和上下文混杂:SDK 重试、框架接管、手动初始化、缓存失效 reload —— 这些信号一旦没对齐,appendChild 就会变成“appendTwice”。排查时别只盯代码行,要盯住每个插入动作背后的触发条件是否唯一。