0

0

如何将 HTML 结构解析为文本与标记分离的嵌套对象数组

花韻仙語

花韻仙語

发布时间:2026-02-28 16:34:01

|

694人浏览过

|

来源于php中文网

原创

如何将 HTML 结构解析为文本与标记分离的嵌套对象数组

本文介绍一种基于递归遍历 DOM 树的专业方法,将任意 HTML 片段准确拆解为按渲染顺序排列的 {text: "..."} 和 {markup: ""} 对象数组,完美处理嵌套、兄弟节点及闭合标签位置问题。

本文介绍一种基于递归遍历 dom 树的专业方法,将任意 html 片段准确拆解为按渲染顺序排列的 `{text: "..."}` 和 `{markup: ""}` 对象数组,完美处理嵌套、兄弟节点及闭合标签位置问题。

在前端开发中,有时需要对 HTML 内容进行语义化结构分析——例如实现富文本编辑器的内容序列化、无障碍辅助解析、或自定义 Markdown/HTML 混合渲染器。核心挑战在于:必须严格保持 DOM 渲染时的节点顺序,同时区分纯文本内容(Text 节点)与 HTML 标记(Element 节点),尤其当元素存在嵌套或相邻兄弟元素时,闭合标签的位置极易出错。

使用 TreeWalker 或线性遍历 childNodes 的迭代逻辑容易陷入边界判断困境(如父元素末尾闭合时机、子元素与文本混排时的插入顺序)。而递归深度优先遍历(DFS)天然契合 DOM 树结构,能自动保证:

  • 开始标签在子内容之前;
  • 所有子节点(含文本与嵌套元素)被完整处理;
  • 结束标签在子内容之后 —— 完全符合 HTML 渲染流。

以下为简洁、健壮、可直接复用的实现方案:

function parseHtmlToTokenArray(rootNode) {
  const tokens = [];

  function walk(node) {
    for (const child of node.childNodes) {
      if (child.nodeType === Node.TEXT_NODE) {
        const text = child.textContent.trim();
        // 可选:跳过纯空白文本(提升结果纯净度)
        if (text.length > 0) {
          tokens.push({ text });
        }
      } else if (child.nodeType === Node.ELEMENT_NODE) {
        // 推入开始标签(小写化确保规范)
        tokens.push({ markup: `<${child.tagName.toLowerCase()}>` });
        // 递归处理所有子节点
        if (child.hasChildNodes()) {
          walk(child);
        }
        // 推入结束标签
        tokens.push({ markup: `</${child.tagName.toLowerCase()}>` });
      }
      // 忽略注释、CDATA 等其他节点类型(如需支持可扩展)
    }
  }

  walk(rootNode);
  return tokens;
}

// 使用示例
const htmlString = `
  <h2 id="mcetoc_1h1m1ll27l">Lorem ipsum dolor sit amet...</h2>
  <p>Lorem ipsum...<a href="#">tr</a><a title="titulo">adsf afjdasi k</a></p><div class="aritcle_card flexRow">
                                                        <div class="artcardd flexRow">
                                                                <a class="aritcle_card_img" href="/ai/1496" title="PhotoAid Image Upscaler"><img
                                                                                src="https://img.php.cn/upload/ai_manual/000/969/633/68b7a456e0374854.png" alt="PhotoAid Image Upscaler"  onerror="this.onerror='';this.src='/static/lhimages/moren/morentu.png'" ></a>
                                                                <div class="aritcle_card_info flexColumn">
                                                                        <a href="/ai/1496" title="PhotoAid Image Upscaler">PhotoAid Image Upscaler</a>
                                                                        <p>PhotoAid出品的免费在线AI图片放大工具</p>
                                                                </div>
                                                                <a href="/ai/1496" title="PhotoAid Image Upscaler" class="aritcle_card_btn flexRow flexcenter"><b></b><span>下载</span> </a>
                                                        </div>
                                                </div><p><span>立即学习</span>“<a href="https://pan.quark.cn/s/cb6835dc7db1" style="text-decoration: underline !important; color: blue; font-weight: bolder;" rel="nofollow" target="_blank">前端免费学习笔记(深入)</a>”;</p>
`;
const tempDiv = document.createElement('div');
tempDiv.innerHTML = htmlString;
const result = parseHtmlToTokenArray(tempDiv);
console.log(result);

输出效果示例(节选):

[
  {"markup": "<h2>"},
  {"text": "Lorem ipsum dolor sit amet..."},
  {"markup": "</h2>"},
  {"markup": "<p>"},
  {"text": "Lorem ipsum..."},
  {"markup": "<a>"},
  {"text": "tr"},
  {"markup": "</a>"},
  {"markup": "<a>"},
  {"text": "adsf afjdasi k"},
  {"markup": "</a>"},
  {"markup": "</p>"}
]

⚠️ 关键注意事项:

  • 空格与换行处理:原始 textContent 包含 HTML 中的空白符(如换行、缩进)。生产环境建议调用 .trim() 或使用 innerText(注意其会触发布局计算);若需保留格式,可改用 nodeValue 并预处理 \n\t\r。
  • 属性完整性:当前方案仅生成基础标签名(如 ),不包含属性。如需还原完整 outerHTML,可替换为 child.outerHTML,但需注意:... 会被整体推入一次,破坏“开–内容–闭”原子结构。此时应先提取起始标签(正则或 child.cloneNode(false).outerHTML),再单独处理闭合。
  • 性能考量:对于超大文档(>10k 节点),递归可能触发栈溢出。可改用显式栈的迭代 DFS(维护 {node, state: 'enter'|'exit'} 元组),但绝大多数 CMS 或编辑器场景无需优化。
  • 安全边界:本函数操作的是已解析的 DOM 节点,不执行 HTML 字符串解析,因此无 XSS 风险;但若输入源自不可信字符串,请务必先通过 DOMPurify.sanitize() 等库净化。

该方案以最小认知负荷达成最高结构保真度——它不试图“修复”DOM,而是忠实反映浏览器引擎的渲染遍历逻辑。掌握此模式,你将能轻松构建 HTML 解析中间层,为内容分析、序列化、差异对比等高阶功能奠定坚实基础。

相关文章

HTML速学教程(入门课程)
HTML速学教程(入门课程)

HTML怎么学习?HTML怎么入门?HTML在哪学?HTML怎么学才快?不用担心,这里为大家提供了HTML速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

638

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

218

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1560

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

645

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1068

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1021

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

186

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

89

2025.08.07

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

0

2026.02.28

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.5万人学习

AngularJS教程
AngularJS教程

共24课时 | 3.9万人学习

CSS教程
CSS教程

共754课时 | 37.1万人学习

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

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