# OCR 结果合并工具说明 ## 概述 本工具包用于合并不同 OCR 工具的识别结果,主要功能是将结构化识别结果(如 MinerU、PaddleOCR_VL)与精确文字框坐标(PaddleOCR)进行合并,生成包含完整 bbox 信息的增强版 Markdown 和 JSON 文件。 ## 支持的合并方式 ### 1. MinerU + PaddleOCR - **输入**: - MinerU 的 JSON 文件(包含表格结构、段落等) - PaddleOCR 的 JSON 文件(包含精确的文字框坐标) - **输出**: - 增强的 Markdown(包含 bbox 注释) - 合并的 JSON(MinerU 格式 + bbox 信息) ### 2. PaddleOCR_VL + PaddleOCR - **输入**: - PaddleOCR_VL 的 JSON 文件(包含版面分析、表格识别等) - PaddleOCR 的 JSON 文件(包含精确的文字框坐标) - **输出**: - 增强的 Markdown(包含 bbox 注释) - 合并的 JSON(转换为 MinerU 格式 + bbox 信息) ## 目录结构 ``` merger/ ├── __init__.py # 包初始化文件 ├── merger_core.py # MinerU 合并器核心类 ├── paddleocr_vl_merger.py # PaddleOCR_VL 合并器核心类 ├── merge_mineru_paddle_ocr.py # MinerU + PaddleOCR 主程序 ├── merge_paddleocr_vl_paddleocr.py # PaddleOCR_VL + PaddleOCR 主程序 ├── text_matcher.py # 文本匹配模块(共用) ├── bbox_extractor.py # Bbox 提取模块(共用) ├── data_processor.py # 数据处理模块(共用) ├── markdown_generator.py # Markdown 生成模块(共用) └── README.md # 本说明文档 ``` ## 核心模块说明 ### 1. 合并器类 #### `MinerUPaddleOCRMerger` (merger_core.py) ```python class MinerUPaddleOCRMerger: """MinerU 和 PaddleOCR 结果合并器""" def __init__(self, look_ahead_window: int = 10, similarity_threshold: int = 90): """ Args: look_ahead_window: 向前查找的窗口大小 similarity_threshold: 文本相似度阈值(0-100) """ def merge_table_with_bbox(self, mineru_json_path: str, paddle_json_path: str) -> List[Dict]: """合并 MinerU 和 PaddleOCR 的结果""" def generate_enhanced_markdown(self, merged_data: List[Dict], output_path: str = None) -> str: """生成增强的 Markdown""" ``` #### `PaddleOCRVLMerger` (paddleocr_vl_merger.py) ```python class PaddleOCRVLMerger: """PaddleOCR_VL 和 PaddleOCR 结果合并器""" def __init__(self, look_ahead_window: int = 10, similarity_threshold: int = 90): """ Args: look_ahead_window: 向前查找的窗口大小 similarity_threshold: 文本相似度阈值(0-100) """ def merge_table_with_bbox(self, paddleocr_vl_json_path: str, paddle_json_path: str) -> List[Dict]: """合并 PaddleOCR_VL 和 PaddleOCR 的结果""" def generate_enhanced_markdown(self, merged_data: List[Dict], output_path: str = None, data_format: str = None) -> str: """生成增强的 Markdown""" ``` ### 2. 共用模块 #### `TextMatcher` (text_matcher.py) 负责文本匹配,找到 MinerU/PaddleOCR_VL 文本在 PaddleOCR 结果中的对应位置。 **核心方法**: ```python def find_matching_bbox(self, target_text: str, text_boxes: List[Dict], start_index: int = 0, last_match_index: int = 0, look_ahead_window: int = 10) -> Tuple[Optional[Dict], int, int]: """ 查找匹配的 bbox Args: target_text: 目标文本 text_boxes: PaddleOCR 文字框列表 start_index: 开始搜索的位置 last_match_index: 上一次匹配的索引 look_ahead_window: 向前查找窗口大小 Returns: (匹配的文字框, 新的开始位置, 新的last_match_index) """ ``` **匹配策略**: 1. **顺序匹配**:优先从 `start_index` 开始顺序查找 2. **窗口回溯**:在 `[last_match_index - look_ahead_window, last_match_index + look_ahead_window]` 范围内查找 3. **相似度计算**:使用 `fuzzywuzzy` 计算文本相似度 4. **指针更新**:匹配成功后更新指针,避免重复匹配 #### `BBoxExtractor` (bbox_extractor.py) 负责从 PaddleOCR 结果中提取文字框信息。 **核心方法**: ```python def extract_paddle_text_boxes(self, paddle_data: Dict) -> List[Dict]: """ 提取 PaddleOCR 的文字框信息 Returns: [ { 'text': '文本内容', 'bbox': [x1, y1, x2, y2], 'score': 0.99, 'used': False }, ... ] """ def extract_table_cells_with_bbox(self, merged_data: List[Dict]) -> List[Dict]: """ 提取所有表格单元格及其 bbox 信息 Returns: [ { 'text': '单元格文本', 'bbox': [x1, y1, x2, y2], 'table_index': 0, 'row': 1, 'col': 2 }, ... ] """ ``` #### `DataProcessor` (data_processor.py) 负责处理 MinerU/PaddleOCR_VL 数据,添加 bbox 信息。 **核心方法**: ```python def process_mineru_data(self, mineru_data: List[Dict], paddle_text_boxes: List[Dict]) -> List[Dict]: """ 处理 MinerU 数据,添加 bbox 信息 处理类型: - text: 普通文本 - title: 标题 - table: 表格 - list: 列表 - image: 图片 - equation: 公式 """ def process_paddleocr_vl_data(self, paddleocr_vl_data: Dict, paddle_text_boxes: List[Dict]) -> List[Dict]: """ 处理 PaddleOCR_VL 数据,添加 bbox 信息 处理类型: - paragraph_title: 段落标题 - figure_title: 图片标题 - text: 文本 - table: 表格 - figure: 图片 - equation: 公式 - reference: 参考文献 """ ``` **表格处理逻辑**: 1. 解析 HTML 表格结构 2. 逐个单元格匹配 PaddleOCR 文字框 3. 为每个 `` 添加 `data-bbox`、`data-paddle-index`、`data-score` 属性 4. 处理合并单元格(colspan、rowspan) #### `MarkdownGenerator` (markdown_generator.py) 负责将合并后的数据生成 Markdown 文件。 **核心特性**: - **自动格式检测**:自动识别 MinerU 或 PaddleOCR_VL 格式 - **bbox 注释**:为每个元素添加 `` 注释 - **表格增强**:保留表格中的 bbox 属性 - **图片处理**:自动复制图片文件 **核心方法**: ```python def detect_data_format(merged_data: List[Dict]) -> str: """ 检测数据格式 Returns: 'mineru' 或 'paddleocr_vl' """ def generate_enhanced_markdown(merged_data: List[Dict], output_path: Optional[str] = None, source_file: Optional[str] = None, data_format: Optional[str] = None) -> str: """ 生成增强的 Markdown Args: merged_data: 合并后的数据 output_path: 输出路径 source_file: 源文件路径(用于复制图片) data_format: 数据格式,None 则自动检测 """ ``` **格式化方法**: - **MinerU 格式**:`_format_mineru_*()` 系列方法 - **PaddleOCR_VL 格式**:`_format_paddleocr_vl_*()` 系列方法 - **通用方法**:`_format_equation()`, `_format_metadata()` 等 ## 使用方法 ### 1. MinerU + PaddleOCR 合并 #### 命令行使用 ```bash # 单文件处理 python merger/merge_mineru_paddle_ocr.py \ --mineru-file /path/to/mineru_page_001.json \ --paddle-file /path/to/paddle_page_001.json \ --output-dir /path/to/output \ --output-type both # 批量处理 python merger/merge_mineru_paddle_ocr.py \ --mineru-dir /path/to/mineru_results \ --paddle-dir /path/to/paddle_results \ --output-dir /path/to/output \ --output-type both \ --window 15 \ --threshold 85 ``` #### 参数说明 | 参数 | 说明 | 默认值 | |------|------|--------| | `--mineru-file` | MinerU JSON 文件路径(单文件模式) | - | | `--paddle-file` | PaddleOCR JSON 文件路径(单文件模式) | - | | `--mineru-dir` | MinerU 结果目录(批量模式) | - | | `--paddle-dir` | PaddleOCR 结果目录(批量模式) | - | | `-o, --output-dir` | 输出目录(必需) | - | | `-f, --output-type` | 输出格式:json/markdown/both | both | | `-w, --window` | 向前查找窗口大小 | 15 | | `-t, --threshold` | 文本相似度阈值(0-100) | 80 | ### 2. PaddleOCR_VL + PaddleOCR 合并 #### 命令行使用 ```bash # 单文件处理 python merger/merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-file /path/to/paddleocr_vl_page_001.json \ --paddle-file /path/to/paddle_page_001.json \ --output-dir /path/to/output \ --output-type both # 批量处理 python merger/merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-dir /path/to/paddleocr_vl_results \ --paddle-dir /path/to/paddle_results \ --output-dir /path/to/output \ --output-type both \ --window 15 \ --threshold 85 ``` #### 参数说明 | 参数 | 说明 | 默认值 | |------|------|--------| | `--paddleocr-vl-file` | PaddleOCR_VL JSON 文件路径(单文件模式) | - | | `--paddle-file` | PaddleOCR JSON 文件路径(单文件模式) | - | | `--paddleocr-vl-dir` | PaddleOCR_VL 结果目录(批量模式) | - | | `--paddle-dir` | PaddleOCR 结果目录(批量模式) | - | | `-o, --output-dir` | 输出目录(必需) | - | | `-f, --output-type` | 输出格式:json/markdown/both | both | | `-w, --window` | 向前查找窗口大小 | 15 | | `-t, --threshold` | 文本相似度阈值(0-100) | 80 | ## 输出格式 ### 1. 增强的 Markdown ```markdown # 账务明细清单 # Statement Of Account 开户银行:呼和浩特成吉思汗大街 ...
日期Date 业务类型Business Type 票据号Bill No.
``` ### 2. 合并的 JSON #### MinerU 格式 ```json [ { "type": "text", "text": "账务明细清单", "text_level": 1, "bbox": [717, 191, 917, 229], "page_idx": 0 }, { "type": "table", "table_body": "...
", "table_body_with_bbox": "...
", "bbox": [176, 406, 1468, 1920], "bbox_mapping": "merged_from_paddle_ocr", "table_cells": [ { "text": "日期Date", "bbox": [183, 413, 293, 438], "paddle_index": 10, "score": 0.9995, "row": 0, "col": 0 } ], "page_idx": 0 } ] ``` #### PaddleOCR_VL 格式(转换为 MinerU 格式) ```json [ { "type": "text", "text": "账务明细清单", "text_level": 1, "bbox": [719, 194, 924, 264], "page_idx": 0 }, { "type": "table", "table_body": "...
", "table_body_with_bbox": "...
", "bbox": [177, 256, 1464, 393], "bbox_mapping": "merged_from_paddle_ocr", "table_cells": [...], "page_idx": 0 } ] ``` ## 核心算法 ### 1. 文本匹配算法 **目标**:在 PaddleOCR 的文字框列表中找到与 MinerU/PaddleOCR_VL 文本最匹配的 bbox。 **策略**: 1. **顺序匹配优先**:从 `start_index` 开始顺序查找 2. **窗口回溯**:如果顺序匹配失败,在上一次匹配位置附近的窗口内查找 3. **相似度筛选**:使用 `fuzzywuzzy.partial_ratio` 计算相似度,阈值默认 80% 4. **指针管理**:维护两个指针 - `start_index`:下次搜索的起始位置 - `last_match_index`:上一次匹配的位置(用于窗口回溯) **伪代码**: ```python def find_matching_bbox(target_text, text_boxes, start_index, last_match_index, window): # 第一阶段:顺序查找 for i in range(start_index, len(text_boxes)): if similarity(target_text, text_boxes[i].text) >= threshold: return text_boxes[i], i + 1, i # 第二阶段:窗口回溯 window_start = max(0, last_match_index - window) window_end = min(len(text_boxes), last_match_index + window) best_match = None best_score = 0 best_index = -1 for i in range(window_start, window_end): score = similarity(target_text, text_boxes[i].text) if score > best_score and score >= threshold: best_match = text_boxes[i] best_score = score best_index = i if best_match: return best_match, start_index, best_index return None, start_index, last_match_index ``` ### 2. 表格单元格匹配算法 **目标**:为表格中的每个单元格找到对应的 PaddleOCR 文字框。 **步骤**: 1. **解析表格 HTML**:使用 BeautifulSoup 解析 `` 结构 2. **逐行逐列处理**: ```python for row in table.find_all('tr'): for cell in row.find_all(['td', 'th']): cell_text = cell.get_text() matched_bbox = find_matching_bbox(cell_text, ...) # 添加属性 cell['data-bbox'] = str(matched_bbox['bbox']) cell['data-paddle-index'] = matched_bbox['index'] cell['data-score'] = matched_bbox['score'] ``` 3. **处理合并单元格**: - 检测 `colspan` 和 `rowspan` 属性 - 为每个展开的单元格使用相同的 bbox 4. **指针更新**:每次匹配成功后更新 `paddle_pointer`,避免重复匹配 ## 参数调优建议 ### 1. `look_ahead_window` (向前查找窗口) **作用**:当顺序匹配失败时,在上一次匹配位置附近的窗口内查找。 **推荐值**: - **顺序文档**(如流水记录):`10-15` - **复杂版面**(如多列排版):`15-25` - **表格密集型**:`5-10` **调优方法**: ```python # 如果发现匹配错位,可以适当增大窗口 merger = MinerUPaddleOCRMerger(look_ahead_window=20) # 如果发现误匹配,可以适当减小窗口 merger = MinerUPaddleOCRMerger(look_ahead_window=8) ``` ### 2. `similarity_threshold` (相似度阈值) **作用**:控制文本匹配的严格程度。 **推荐值**: - **高质量 OCR**(如扫描件):`85-90` - **一般质量**(如拍照):`75-85` - **低质量**(如模糊图片):`70-80` **调优方法**: ```python # 如果发现漏匹配,可以降低阈值 merger = MinerUPaddleOCRMerger(similarity_threshold=75) # 如果发现误匹配,可以提高阈值 merger = MinerUPaddleOCRMerger(similarity_threshold=90) ``` ## 开发指南 ### 扩展新的合并器 1. **创建新的合并器类**: ```python # merger/my_custom_merger.py from .text_matcher import TextMatcher from .data_processor import DataProcessor class MyCustomMerger: def __init__(self): self.text_matcher = TextMatcher() self.data_processor = DataProcessor(self.text_matcher) def merge_data(self, custom_json, paddle_json): # 实现自定义合并逻辑 pass ``` 2. **在 DataProcessor 中添加处理方法**: ```python # merger/data_processor.py def process_my_custom_data(self, custom_data, paddle_text_boxes): # 处理自定义格式数据 pass ``` 3. **在 MarkdownGenerator 中添加格式化方法**: ```python # merger/markdown_generator.py @staticmethod def _format_my_custom_element(item: Dict) -> List[str]: # 格式化自定义元素 pass ``` ### 单元测试 ```python # tests/test_merger.py import pytest from merger import MinerUPaddleOCRMerger def test_text_matching(): merger = MinerUPaddleOCRMerger() # 测试文本匹配逻辑 def test_table_processing(): merger = MinerUPaddleOCRMerger() # 测试表格处理逻辑 ``` ## 版本历史 ### v1.2.0 (当前版本) - ✅ 新增 PaddleOCR_VL 合并支持 - ✅ 重构为模块化架构,提高代码复用率 - ✅ 新增自动格式检测功能 - ✅ 优化文本匹配算法 - ✅ 改进表格单元格 bbox 匹配 ### v1.1.0 - ✅ 优化表格合并单元格处理 - ✅ 新增批量处理模式 - ✅ 改进窗口回溯算法 ### v1.0.0 - ✅ 初始版本 - ✅ 支持 MinerU + PaddleOCR 合并 - ✅ 基本的文本匹配功能 --- ### 运行试验数据 #### 1. mineru-vlm-2.5.3 ```bash echo "A用户_单元格扫描流水" python merge_mineru_paddle_ocr.py \ --mineru-dir "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/mineru-vlm-2.5.3_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/merged_results" \ --output-type "both" echo "B用户_扫描流水" python merge_mineru_paddle_ocr.py \ --mineru-dir "/Users/zhch158/workspace/data/流水分析/B用户_扫描流水/mineru-vlm-2.5.3_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/B用户_扫描流水/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/B用户_扫描流水/merged_results" \ --output-type "both" echo "德_内蒙古银行照" python merge_mineru_paddle_ocr.py \ --mineru-dir "/Users/zhch158/workspace/data/流水分析/德_内蒙古银行照/mineru-vlm-2.5.3_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/德_内蒙古银行照/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/德_内蒙古银行照/merged_results" \ --output-type "both" echo "对公_招商银行图" python merge_mineru_paddle_ocr.py \ --mineru-dir "/Users/zhch158/workspace/data/流水分析/对公_招商银行图/mineru-vlm-2.5.3_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/对公_招商银行图/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/对公_招商银行图/merged_results" \ --output-type "both" echo "至远彩色印刷工业有限公司" python merge_mineru_paddle_ocr.py \ --mineru-dir "/Users/zhch158/workspace/data/至远彩色印刷工业有限公司/mineru-vlm-2.5.3_Results" \ --paddle-dir "/Users/zhch158/workspace/data/至远彩色印刷工业有限公司/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/至远彩色印刷工业有限公司/merged_results" \ --output-type "both" ``` #### 2. PaddleOCR_VL_Results ```bash echo "A用户_单元格扫描流水" python merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-dir "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/PaddleOCR_VL_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/PaddleOCR_VL_Results_cell_bbox" \ --output-type "both" echo "B用户_扫描流水" python merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-dir "/Users/zhch158/workspace/data/流水分析/B用户_扫描流水/PaddleOCR_VL_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/B用户_扫描流水/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/B用户_扫描流水/PaddleOCR_VL_Results_cell_bbox" \ --output-type "both" echo "德_内蒙古银行照" python merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-dir "/Users/zhch158/workspace/data/流水分析/德_内蒙古银行照/PaddleOCR_VL_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/德_内蒙古银行照/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/德_内蒙古银行照/PaddleOCR_VL_Results_cell_bbox" \ --output-type "both" echo "对公_招商银行图" python merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-dir "/Users/zhch158/workspace/data/流水分析/对公_招商银行图/PaddleOCR_VL_Results" \ --paddle-dir "/Users/zhch158/workspace/data/流水分析/对公_招商银行图/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/流水分析/对公_招商银行图/PaddleOCR_VL_Results_cell_bbox" \ --output-type "both" echo "至远彩色印刷工业有限公司" python merge_paddleocr_vl_paddleocr.py \ --paddleocr-vl-dir "/Users/zhch158/workspace/data/至远彩色印刷工业有限公司/PaddleOCR_VL_Results" \ --paddle-dir "/Users/zhch158/workspace/data/至远彩色印刷工业有限公司/data_PPStructureV3_Results" \ --output-dir "/Users/zhch158/workspace/data/至远彩色印刷工业有限公司/PaddleOCR_VL_Results_cell_bbox" \ --output-type "both" ```