在 HTML 中,仅修改已存在 <script type="text/skip-hydration"> 元素的 type 属性为 "module" 并不能触发浏览器加载与执行;必须用新创建的、类型正确的 <script type="module"> 元素显式替换原节点。
在 html 中,仅修改已存在 `<script type="text/skip-hydration">` 元素的 `type` 属性为 `"module"` 并不能触发浏览器加载与执行;必须用新创建的、类型正确的 `<script type="module">` 元素显式替换原节点。</script>
浏览器对 <script> 标签的解析和执行行为在元素插入 DOM 时即已确定。一旦 <script> 节点被解析(例如页面初始加载阶段),后续通过 setAttribute('type', 'module') 修改其 type 属性不会重新触发模块加载机制——这是因为模块脚本的加载、解析和执行流程由浏览器在节点挂载时同步判定,属性变更属于运行时操作,不触发重解析。
因此,简单地调用 script.setAttribute('type', 'module') 或 script.type = 'module' 是无效的;而直接 appendChild(script) 也存在风险:若该 script 节点已存在于 DOM 中(如在 <head> 或 <body> 内),重复 append 会引发移动(move)而非重载,且可能因执行时机问题导致模块未按预期初始化(尤其当 src 指向外部文件时)。
✅ 正确做法是:克隆原节点 → 修改克隆体的 type → 用克隆体原位替换原节点。这确保了浏览器将新节点视为一个“全新”的模块脚本,并按标准模块流程加载与执行。
以下为推荐实现(兼容现代浏览器):
立即学习“前端免费学习笔记(深入)”;
document.querySelectorAll('script[type="text/skip-hydration"]').forEach(script => { const newScript = script.cloneNode(true); // deep clone to preserve content & attributes newScript.type = 'module'; script.replaceWith(newScript);});
✅ cloneNode(true) 确保内联脚本内容(如 <script type="text/skip-hydration">console.log('hi');</script>)也被保留;若仅为外链脚本(如 src="runtime.js"),true/false 效果一致,但建议统一使用 true 以保持健壮性。
✅ replaceWith() 是标准 DOM 方法,语义清晰、原子性强,避免手动 remove() + insertBefore() 的冗余操作。
更简洁的写法(利用 Object.assign):
document.querySelectorAll('script[type="text/skip-hydration"]') .forEach(script => script.replaceWith(Object.assign(script.cloneNode(true), { type: 'module' })) );
⚠️ 注意事项:
总结:DOM 节点的脚本类型不可热更新,必须通过“销毁-重建-替换”完成语义切换。掌握 cloneNode + replaceWith 这一模式,是安全、可预测地实现脚本类型动态启用的关键。