news 2026/3/25 0:32:16

RAG基础:基于markdown_split的Markdown文本分割实战

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
RAG基础:基于markdown_split的Markdown文本分割实战

一、要求实现如下功能:

功能:

  • headers_to_split_on

  • 支出混合的拆块(支持 chunk_size、chunk_overlap)

  • 支持 chunk 元数据

  • markdown 中的代码不拆分

  • 支持strip_headers 参数设置

二、题解

思路解析:

实现思路
1. 分层处理策略 :
- 第一层:使用 MarkdownHeaderTextSplitter 按标题级别分割文本,保留标题结构
- 第二层:对非代码块内容使用 RecursiveCharacterTextSplitter 进行语义分割
- 特殊处理:识别并完整保留代码块,不进行拆分

2. 核心功能实现 :
- 标题分割 :通过 headers_to_split_on 参数定义要分割的标题级别
- 混合拆块 :结合标题分割和字符分割,支持 chunk_size 和 chunk_overlap 参数
- 元数据保留 :在分割过程中传递和保留文档元数据
- 代码块保护 :通过检测代码块标记(```),确保代码块完整性

3. 技术要点 :
- 使用状态机识别代码块的开始和结束
- 对普通文本和代码块采用不同的处理策略
- 保留原始文档的元数据信息
- 支持自定义标题级别和分割参数

代码:

from langchain_text_splitters import MarkdownHeaderTextSplitter,RecursiveCharacterTextSplitter def markdown_split( markdown_text, # 输入的Markdown文本 headers_to_split_on=None, # 要分割的标题级别 chunk_size=500, # 单个块最大字符数 chunk_overlap=100, # 相邻块重叠字符数 code_block_handling=True # 是否保留代码块完整性 ): """ Markdown文本分割函数 Args: markdown_text: 输入的Markdown文本 headers_to_split_on: 要分割的标题级别,格式为[(\"#\", \"一级标题\"), (\"##\", \"二级标题\")] chunk_size: 单个块最大字符数 chunk_overlap: 相邻块重叠字符数 code_block_handling: 是否保留代码块完整性 Returns: 分割后的Document对象列表 """ # 默认标题级别 if headers_to_split_on is None: headers_to_split_on = [ ("#", "一级标题"), ("##", "二级标题"), ("###", "三级标题"), ] # 1. 使用MarkdownHeaderTextSplitter按标题分割 markdown_splitter = MarkdownHeaderTextSplitter( headers_to_split_on=headers_to_split_on, strip_headers=False, return_each_line=False ) # 执行标题分割 header_split_docs = markdown_splitter.split_text(markdown_text) # 2. 初始化递归字符分割器,用于二次分割长内容 text_splitter = RecursiveCharacterTextSplitter( chunk_size=chunk_size, chunk_overlap=chunk_overlap, separators=["\n\n", "\n", "。", "!", "?", ";", ",", " "], length_function=len, is_separator_regex=False ) # 3. 处理每个标题分割后的部分 final_docs = [] for doc in header_split_docs: content = doc.page_content metadata = doc.metadata.copy() # 如果需要保留代码块完整性 if code_block_handling and "```" in content: # 分割代码块和普通文本 parts = [] # current_part = "" in_code_block = False for line in content.split("\n"): if line.startswith("```"): if in_code_block: # 代码块结束 current_part += line + "\n" parts.append((current_part, True)) # True表示是代码块 current_part = "" in_code_block = False else: # 代码块开始 if current_part: parts.append((current_part, False)) # False表示普通文本 current_part = line + "\n" in_code_block = True else: current_part += line + "\n" # 处理最后一个部分 if current_part: parts.append((current_part, False)) # 对普通文本进行分割,保留代码块完整 for part, is_code in parts: if is_code: # 代码块直接添加,不分割 final_docs.append(type(doc)(page_content=part, metadata=metadata)) else: # 普通文本使用递归分割器 sub_docs = text_splitter.create_documents([part]) for sub_doc in sub_docs: # 保留原始元数据 sub_doc.metadata = metadata.copy() final_docs.append(sub_doc) else: # 不需要保留代码块完整性,直接使用递归分割器 sub_docs = text_splitter.create_documents([content]) for sub_doc in sub_docs: # 保留原始元数据 sub_doc.metadata = metadata.copy() final_docs.append(sub_doc) return final_docs with open("测试数据.md",'r',encoding='utf-8')as f: test_markdown=f.read() # 执行分割 result_docs = markdown_split( test_markdown, chunk_size=200, chunk_overlap=50 ) # 输出结果 print("=== markdown_split 分割结果 ===") for i, doc in enumerate(result_docs, 1): print(f"\n块 {i}:") print(f"字符数: {len(doc.page_content)}") print(f"元数据: {doc.metadata}") print(f"内容:\n{doc.page_content}") print("-" * 80)

运行结果:

数据样例已上传。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/21 15:39:21

金属检测机的核心原理与技术指标解析

放在现代工业生产里,金属检测机是质量控制设备里不能少的,它核心的原理是基于电磁感应技术。产品含有金属异物,通过检测区域时,会让交变磁场有扰动,设备内部接收线圈捕捉信号变化,经过电路分析处理&#xf…

作者头像 李华
网站建设 2026/3/20 6:30:00

智慧城市、能源等优质学术会议分享!

​ ↑↑↑ 了解更多详细会议信息、投稿优惠 请添加会议老师 2026年可持续发展与城市规划国际学术会议(SDUP 2026) 2026 International Conference on Sustainable Development and Urban Planning ​ ↑↑↑ 了解更多详细会议信息、投稿优惠 请添…

作者头像 李华
网站建设 2026/3/22 18:23:08

一张图看懂网络空间安全:从网络层到应用层的“防护圈”都有哪些?

什么是网络安全,网络空间安全有哪些安全? 本文章详细列举出网络空间安全的十六大种类 网络空间安全是一个覆盖 “物理层 - 网络层 - 应用层 - 数据层 - 业务层” 的全域防护体系,其安全种类可根据防护对象、技术场景和业务领域划分为 16 大…

作者头像 李华
网站建设 2026/3/15 7:27:47

如何从Target平台获取搜索列表数据的API接口

在现代Web开发中,API(应用程序接口)是实现平台数据交互的核心工具。本文将以Target平台为例,详细介绍如何通过其API接口获取搜索列表数据。Target平台提供了一个RESTful API,允许开发者查询关键词相关的搜索结果&#…

作者头像 李华
网站建设 2026/3/15 7:32:10

Apple生态自动化理想之选!亚马逊云科技Mac实例一键部署OpenClaw

企业或团队真正需要的,不是一个“会聊天的AI玩具”,而是一个能在云端持续工作、每个操作都留痕可查、无缝融入现有沟通工具的生产力助手。OpenClaw(原名Clawdbot、Moltbot)火了之后,大家发现:部署方式直接决…

作者头像 李华