0

0

解决JSDOM中MutationObserver的“参数1不是Node类型”错误

霞舞

霞舞

发布时间:2025-07-16 17:26:25

|

456人浏览过

|

来源于php中文网

原创

解决jsdom中mutationobserver的“参数1不是node类型”错误

正如摘要所述,当你在JSDOM中使用MutationObserver时,可能会遇到一个令人困惑的错误:“TypeError: Failed to execute 'observe' on 'MutationObserver': parameter 1 is not of type 'Node'”。 即使你确信传递给observe方法的参数是一个有效的DOM节点,该错误仍然可能出现。这通常是因为存在多个JSDOM实例,导致MutationObserver和DOM节点来自不同的JSDOM环境。

问题根源:多个JSDOM实例

在Jest测试环境中,jest-environment-jsdom 会自动创建一个全局的JSDOM实例。同时,你的测试代码可能又显式地创建了另一个JSDOM实例(例如,通过 new JSDOM())。

当你在测试中使用 new MutationObserver() 时,它会默认使用全局的JSDOM实例。但是,你传递给 observe 方法的DOM节点可能来自你显式创建的JSDOM实例。由于JSDOM不允许跨实例操作节点,因此会抛出上述错误。

解决方案:确保MutationObserver和DOM节点来自同一实例

解决这个问题的关键在于确保你使用的 MutationObserver 和DOM节点都来自同一个JSDOM实例。以下是几种可能的解决方案:

1. 使用JSDOM实例的MutationObserver

最直接的解决方案是将 MutationObserver 的创建与特定的JSDOM实例关联起来。例如,如果你创建了一个名为 dom 的JSDOM实例,那么你应该使用 new dom.window.MutationObserver() 来创建 MutationObserver。

修改后的代码示例:

import { JSDOM } from 'jsdom';
import fs from 'fs';
import path from 'path';

const html = fs.readFileSync(path.resolve(__dirname, '../index.html'), 'utf8');

let dom;
let container;

function waitForElm(dom, parentElm, selector) { // 传递dom实例
  return new Promise(resolve => {
    if (parentElm.querySelector(selector)) {
      return resolve(parentElm.querySelector(selector));
    }

    const observer = new dom.window.MutationObserver(mutations => { // 使用dom实例的MutationObserver
      if (parentElm.querySelector(selector)) {
        resolve(parentElm.querySelector(selector));
        observer.disconnect();
      }
    });

    observer.observe(parentElm, {
      childList: true,
      subtree: true
    });
  });
}

describe('index.html', () => {
  beforeEach(() => {
    dom = new JSDOM(html, { runScripts: 'dangerously', resources: "usable" });
  })

  it('adds a tag', async () => {
    container = dom.window.document.body.querySelector("#app-root");
    const elm = await waitForElm(dom, container, '.my-embed'); // 传递dom实例
    expect(container.querySelector('.my-embed')).not.toBeNull();
  });
});

在这个示例中,我们将 dom 实例传递给 waitForElm 函数,并在 waitForElm 函数中使用 dom.window.MutationObserver 创建 MutationObserver。这样可以确保 MutationObserver 和 parentElm 都来自同一个JSDOM实例。

Proface Avatarize
Proface Avatarize

一个利用AI技术提供高质量专业头像和头像的工具

下载

注意事项:

  • 这种方法需要你将JSDOM实例(或者至少是它的 window 对象)传递给所有需要使用 MutationObserver 的函数。

2. 使用全局的JSDOM实例

另一种解决方案是始终使用全局的JSDOM实例,并避免显式创建新的JSDOM实例。你可以通过 document 对象访问全局JSDOM实例。

修改后的代码示例:

import '@testing-library/jest-dom/extend-expect';
import fs from 'fs';
import path from 'path';

const html = fs.readFileSync(path.resolve(__dirname, '../index.html'), 'utf8');

//  不再显式创建 JSDOM 实例
// let dom;
let container;

function waitForElm(parentElm, selector) {
  return new Promise(resolve => {
    if (parentElm.querySelector(selector)) {
      return resolve(parentElm.querySelector(selector));
    }

    const observer = new MutationObserver(mutations => {
      if (parentElm.querySelector(selector)) {
        resolve(parentElm.querySelector(selector));
        observer.disconnect();
      }
    });

    observer.observe(parentElm, {
      childList: true,
      subtree: true
    });
  });
}

describe('index.html', () => {
  beforeEach(() => {
    //不再使用 new JSDOM,直接修改全局 document
    document.documentElement.innerHTML = html;
    // 或者使用 document.write(html);
    // dom = new JSDOM(html, { runScripts: 'dangerously', resources: "usable" });
  })

  it('adds a tag', async () => {
    container = document.body.querySelector("#app-root");
    const elm = await waitForElm(container, '.my-embed');
    expect(container.querySelector('.my-embed')).not.toBeNull();
  });
});

在这个示例中,我们不再显式创建JSDOM实例,而是直接修改全局的 document 对象的内容。 我们使用document.documentElement.innerHTML = html; 将HTML内容写入全局的JSDOM实例中。 这样,所有的DOM操作都将在同一个JSDOM实例中进行。

注意事项:

  • 这种方法可能会影响其他测试用例,因为你正在修改全局状态。 因此,在使用这种方法时,请确保在每个测试用例之前重置全局状态。
  • 使用 document.write() 或 document.documentElement.innerHTML = ... 设置HTML内容可能会导致一些副作用,例如重新加载脚本。你需要仔细评估这些副作用是否会影响你的测试结果。

总结

MutationObserver 在 JSDOM 中报错“参数1不是Node类型”通常是由于多个JSDOM实例引起的。为了解决这个问题,你需要确保 MutationObserver 和 DOM 节点都来自同一个 JSDOM 实例。 你可以选择使用 JSDOM 实例的 MutationObserver,或者始终使用全局的 JSDOM 实例。 选择哪种方法取决于你的具体需求和测试环境。 在选择解决方案时,请仔细评估每种方法的优缺点,并选择最适合你的情况的方法。

相关专题

更多
html版权符号
html版权符号

html版权符号是“©”,可以在html源文件中直接输入或者从word中复制粘贴过来,php中文网还为大家带来html的相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

591

2023.06.14

html在线编辑器
html在线编辑器

html在线编辑器是用于在线编辑的工具,编辑的内容是基于HTML的文档。它经常被应用于留言板留言、论坛发贴、Blog编写日志或等需要用户输入普通HTML的地方,是Web应用的常用模块之一。php中文网为大家带来了html在线编辑器的相关教程、以及相关文章等内容,供大家免费下载使用。

638

2023.06.21

html网页制作
html网页制作

html网页制作是指使用超文本标记语言来设计和创建网页的过程,html是一种标记语言,它使用标记来描述文档结构和语义,并定义了网页中的各种元素和内容的呈现方式。本专题为大家提供html网页制作的相关的文章、下载、课程内容,供大家免费下载体验。

458

2023.07.31

html空格
html空格

html空格是一种用于在网页中添加间隔和对齐文本的特殊字符,被用于在网页中插入额外的空间,以改变元素之间的排列和对齐方式。本专题为大家提供html空格的相关的文章、下载、课程内容,供大家免费下载体验。

240

2023.08.01

html是什么
html是什么

HTML是一种标准标记语言,用于创建和呈现网页的结构和内容,是互联网发展的基石,为网页开发提供了丰富的功能和灵活性。本专题为大家提供html相关的各种文章、以及下载和课程。

2850

2023.08.11

html字体大小怎么设置
html字体大小怎么设置

在网页设计中,字体大小的选择是至关重要的。合理的字体大小不仅可以提升网页的可读性,还能够影响用户对网页整体布局的感知。php中文网将介绍一些常用的方法和技巧,帮助您在HTML中设置合适的字体大小。

500

2023.08.11

html转txt
html转txt

html转txt的方法有使用文本编辑器、使用在线转换工具和使用Python编程。本专题为大家提供html转txt相关的文章、下载、课程内容,供大家免费下载体验。

306

2023.08.31

html文本框代码怎么写
html文本框代码怎么写

html文本框代码:1、单行文本框【<input type="text" style="height:..;width:..;" />】;2、多行文本框【textarea style=";height:;"></textare】。

417

2023.09.01

excel制作动态图表教程
excel制作动态图表教程

本专题整合了excel制作动态图表相关教程,阅读专题下面的文章了解更多详细教程。

30

2025.12.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号