使用 Python 递归处理结构化文本文件并提取特定数据

使用 Python 递归处理结构化文本文件并提取特定数据

本文详细介绍了如何使用 Python 递归遍历目录下的 .txt 文件,并高效地解析其中结构化的网络速度数据。通过将文件内容按固定行数分块处理,并结合专门的函数进行数据提取、类型转换和格式化输出,实现对下载和上传速度的精确分析与报告。该方法适用于处理具有一致内部结构的大量文本文件。

1. 问题背景与分析

在日常的数据处理任务中,我们经常需要从大量具有相似结构但内容各异的文本文件中提取特定信息。本教程将以一个具体场景为例:从一个包含多层子目录的系统中,查找所有 .txt 文件。每个 .txt 文件都包含两个逻辑部分,每个部分由固定数量的行组成,并以特定标识符(例如 >)分隔。我们的目标是从每个部分的特定行中,提取“下载速度”和“上传速度”的数据,并根据其数值进行条件判断和格式化输出。

示例文件结构如下,其中每个逻辑部分(由 > 引导)固定包含 8 行:

> this is first output and some another contentsthese aresome test linesto fill the fileTesting download speedDownload: 0.00 Mbit/sTesting upload speedUpload: 0.00 Mbit/s> this is second output but other texts go here toothese aresome test linesto fill the fileTesting download speedDownload: 1200.58 Mbit/sTesting upload speedUpload: 857.25 Kbit/s

我们需要对每个文件的每个部分的“Download:”和“Upload:”行进行解析,提取速度值和单位,并根据预设的条件(例如速度为零、小于 600 Mbit/s 等)输出不同的信息。

2. 解决方案概述

为了高效且健壮地解决此类问题,我们采用以下策略:

递归文件查找: 使用 pathlib 库递归地查找所有目标文件。文件内容分块: 由于文件结构一致,每个逻辑部分行数固定,我们可以将文件内容按固定大小分块,将每个逻辑部分视为一个独立的数据块。数据提取与转换: 编写辅助函数,从指定行中解析出速度值(浮点数)和单位(字符串)。结果格式化: 编写辅助函数,根据提取的速度值和单位,生成符合要求的输出字符串。模块化设计: 将不同的功能封装在独立的函数中,提高代码的可读性和可维护性。

3. 具体实现步骤

3.1 定义常量与文件查找

首先,我们需要定义文件中每个逻辑部分的行数以及文件包含的逻辑部分数量。然后,使用 pathlib 库进行递归文件查找。

立即学习“Python免费学习笔记(深入)”;

import sysfrom pathlib import Path# 定义常量:每个逻辑部分的行数,以及文件包含的逻辑部分数量LINES_PER_PART = 8PARTS_PER_FILE = 2 # 根据示例文件,每个文件有两部分def main():    # 递归查找当前目录及其子目录下所有 .txt 文件    result = list(Path(".").rglob("*.txt"))    for filename in result:        with open(filename, 'r') as file:            # 读取文件所有行            lines = file.readlines()            # ... 后续处理 ...

3.2 文件内容分块函数

为了将文件的所有行按固定大小(LINES_PER_PART)分割成多个逻辑部分,我们定义一个 chunks 函数。

def chunks(arr, chunk_size):    """    将列表 arr 分割成大小为 chunk_size 的块。    """    result = []    for i in range(0, len(arr), chunk_size):        result.append(arr[i:i+chunk_size])    return result

在 main 函数中调用此函数:

# ... (在 main 函数内部)            lines = file.readlines()            parts = chunks(lines, LINES_PER_PART) # 将文件内容分块# ...

3.3 速度信息解析函数

parse_speed_info 函数负责从包含速度信息的字符串中提取数值和单位。它假定速度值是字符串的第二个单词,单位是第三个单词。

def parse_speed_info(string):    """    从速度信息字符串中解析出速度值(浮点数)和单位。    示例输入: "Download: 1200.58 Mbit/sn"    示例输出: (1200.58, "Mbit/s")    """    # 忽略 "Download:" 或 "Upload:" 部分,从第二个单词开始解析    speed_info_list = string.split()[1::]    # 返回速度值(转换为浮点数)和单位    return (        float(speed_info_list[0]),        speed_info_list[1].strip() # 移除单位末尾的换行符    )

3.4 速度信息格式化函数

stringify_speed_info 函数根据解析出的速度值和单位,生成符合特定条件(如零、小于 600)的报告字符串。

def stringify_speed_info(speed, unit):    """    根据速度值和单位,生成格式化的输出字符串。    """    if speed == 0:        return "zero"    elif unit == "Mbit/s" and speed < 600.0: # 仅对 Mbit/s 单位进行小于600的判断        return f"less than 600 {unit}"    else:        return f"{speed} {unit}"

注意: 原始问题中对“小于 600”的判断仅针对 Mbit/s 单位,此处已在代码中体现。

3.5 遍历分块并输出结果

在 main 函数中,遍历 parts 列表,对每个逻辑部分提取并处理下载和上传速度信息。根据示例文件结构,下载速度信息在倒数第 3 行 (part[-3]),上传速度信息在最后一行 (part[-1])。

# ... (在 main 函数内部)            parts = chunks(lines, LINES_PER_PART)            for i, part in enumerate(parts, 1): # i 从 1 开始计数,表示第几部分                # 下载速度信息在当前部分的倒数第三行                download_info = parse_speed_info(part[-3])                # 上传速度信息在当前部分的最后一行                upload_info = parse_speed_info(part[-1])                # 打印格式化后的结果                print(f"Download{i} speed of {filename} is {stringify_speed_info(*download_info)}.")                print(f"Upload{i} speed of {filename} is {stringify_speed_info(*upload_info)}.")                print() # 每处理完一个文件的一部分后打印空行# ...

3.6 完整代码

将上述所有部分整合,形成完整的 Python 脚本。

#!/usr/bin/python3from pathlib import Path# 定义常量:每个逻辑部分的行数LINES_PER_PART = 8def chunks(arr, chunk_size):    """    将列表 arr 分割成大小为 chunk_size 的块。    """    result = []    for i in range(0, len(arr), chunk_size):        result.append(arr[i:i+chunk_size])    return resultdef parse_speed_info(string):    """    从速度信息字符串中解析出速度值(浮点数)和单位。    示例输入: "Download: 1200.58 Mbit/sn"    示例输出: (1200.58, "Mbit/s")    """    speed_info_list = string.split()[1::] # 忽略 "Download:" 或 "Upload:" 部分    return (        float(speed_info_list[0]),        speed_info_list[1].strip() # 移除单位末尾的换行符    )def stringify_speed_info(speed, unit):    """    根据速度值和单位,生成格式化的输出字符串。    """    if speed == 0:        return "zero"    elif unit == "Mbit/s" and speed < 600.0:        return f"less than 600 {unit}"    else:        return f"{speed} {unit}"def main():    """    主函数:执行文件查找、解析和结果输出。    """    # 递归查找当前目录及其子目录下所有 .txt 文件    result = list(Path(".").rglob("*.txt"))    for filename in result:        try:            with open(filename, 'r') as file:                lines = file.readlines()                # 将文件内容按固定行数分块                parts = chunks(lines, LINES_PER_PART)                # 遍历每个逻辑部分                for i, part in enumerate(parts, 1):                    # 检查部分行数是否符合预期,避免索引错误                    if len(part) != LINES_PER_PART:                        print(f"Warning: File {filename}, Part {i} has unexpected line count ({len(part)} lines). Skipping.")                        continue                    # 下载速度信息在当前部分的倒数第三行                    download_info = parse_speed_info(part[-3])                    # 上传速度信息在当前部分的最后一行                    upload_info = parse_speed_info(part[-1])                    # 打印格式化后的结果                    print(f"Download{i} speed of {filename} is {stringify_speed_info(*download_info)}.")                    print(f"Upload{i} speed of {filename} is {stringify_speed_info(*upload_info)}.")                    print() # 每处理完一个文件的一部分后打印空行        except Exception as e:            print(f"Error processing file {filename}: {e}", file=sys.stderr)if __name__ == "__main__":    main()

4. 注意事项与总结

文件结构一致性: 本方案的核心假设是所有 .txt 文件都具有严格一致的内部结构,即每个逻辑部分都包含固定数量的行(LINES_PER_PART),并且速度信息始终位于这些部分的固定相对位置。如果文件结构不一致,例如行数不固定或信息位置变化,则需要更复杂的解析逻辑(如基于正则表达式或关键词搜索)。错误处理: 在完整代码中加入了简单的 try-except 块来捕获文件打开或处理过程中的异常,并增加了对 part 长度的检查,以提高程序的健壮性。在实际应用中,可以根据需求增加更详细的错误日志记录和处理机制。内存效率: 对于非常大的文件,file.readlines() 会一次性将所有内容加载到内存中。如果文件大小达到 GB 级别,这可能会导致内存溢出。对于此类情况,可以考虑逐行读取文件,并使用一个缓冲区来构建每个 part,或者使用生成器(yield)来优化 chunks 函数,实现惰性加载。但对于本例中每个文件只有两部分、每部分 8 行的场景,readlines() 是完全可接受的。可扩展性: 通过将解析和格式化逻辑封装在单独的函数中,代码的可读性和可维护性得到了显著提升。如果未来需要提取其他类型的数据或改变输出格式,只需修改相应的函数即可,而无需改动主处理逻辑。

通过上述方法,我们能够高效、准确地从结构化文本文件中提取所需数据,并根据业务逻辑进行灵活的展示。这种分而治之、模块化设计的思想在处理复杂数据解析任务时尤为重要。

以上就是使用 Python 递归处理结构化文本文件并提取特定数据的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1364739.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月14日 04:11:50
下一篇 2025年12月14日 04:11:59

相关推荐

  • Tkinter中为单个字符设置不同字体大小的实现方法

    本教程探讨了在Tkinter的Label或Button组件中为不同字符设置不同字体大小的挑战与解决方案。由于Tkinter的Label和Button组件本身不支持多字体样式,文章将详细介绍如何利用Frame容器结合多个Label组件,并通过布局管理器实现字符的精确排布与对齐,从而达到视觉上的多字体效…

    好文分享 2025年12月14日
    000
  • Python教程:递归解析分段日志文件中的性能数据

    本教程详细介绍了如何使用Python递归地遍历目录,解析结构化文本文件中的分段性能数据。通过定义固定行数的块,提取下载和上传速度信息,并根据预设条件进行格式化输出。内容涵盖文件查找、数据分块、信息提取与自定义格式化函数,适用于处理具有一致内部结构的多部分日志或报告文件,提供了一种高效、可维护的数据处…

    2025年12月14日
    000
  • Python中如何操作ODT文档?odfpy库指南

    python中操作odt文档的核心工具是odfpy库,1. 它允许直接与odf文档的底层xml结构交互,适用于创建、读取、修改和内容提取;2. 使用前需安装odfpy并通过理解odf规范或习惯操作xml节点来构建文档;3. 创建文档时通过添加标题和段落等元素并保存;4. 读取文档时遍历段落和标题获取…

    2025年12月14日 好文分享
    000
  • Edge-TTS 使用中 UnboundLocalError 错误排查与解决

    在使用 Edge-TTS 库时,可能会遇到 `UnboundLocalError: cannot access local variable ‘audio_segment’ where it is not associated with a value` 错误。本文旨在分析该…

    2025年12月14日
    000
  • Python如何实现自动化测试?Selenium与Pytest结合指南

    python实现自动化测试的核心方案是结合selenium和pytest。1. 首先,安装python及相关库(selenium、pytest)并配置浏览器驱动;2. 接着,编写测试脚本,使用selenium模拟用户操作,通过pytest管理测试流程及断言;3. 然后,采用page object m…

    2025年12月14日 好文分享
    000
  • 如何用Python开发网络监控?socket编程

    用python做网络监控基础功能可通过socket编程实现。1.监听本地端口可检测连接请求,通过绑定ip和端口并监听,打印连接来源;2.主动探测远程主机是否在线,尝试建立连接并根据结果判断目标状态,适合批量检测服务器;3.权限、超时设置、并发处理及安全建议是使用中需注意的要点,如低端口绑定需管理员权…

    2025年12月14日 好文分享
    000
  • 怎样用Python处理地理数据?Geopandas库完整操作指南

    geopandas是地理数据处理首选,因它整合了shapely、fiona、matplotlib和pandas功能于一体。1. 它基于pandas dataframe扩展出geodataframe和geoseries,支持空间数据操作;2. 提供统一api简化从加载、清洗到分析、可视化的流程;3. …

    2025年12月14日 好文分享
    000
  • 解决 Edge-TTS 调用中的 UnboundLocalError 错误

    本文旨在解决在使用 Edge-TTS 库进行文本转语音转换时,遇到的 `UnboundLocalError: cannot access local variable ‘audio_segment’ where it is not associated with a valu…

    2025年12月14日
    000
  • 如何使用Python灵活处理不同数量变量的配置校准问题

    本文旨在提供一种优雅的Python编程方法,以应对参数校准过程中变量数量变化的问题。通过引入参数掩码和动态函数构建,可以避免编写大量重复代码,从而实现代码的简洁性和可维护性。我们将通过一个具体的校准案例,展示如何利用这些技术来构建灵活的配置方案。 在参数校准过程中,经常会遇到需要调整的参数数量不确定…

    2025年12月14日
    000
  • 使用可变数量参数进行代码配置的结构化方法

    本文旨在提供一种结构化的方法,用于处理在代码校准过程中参数数量变化的问题。通过引入掩码机制和动态参数传递,避免编写大量重复代码,实现灵活且可维护的参数校准方案。我们将探讨如何利用Python的特性,编写通用的导数计算和校准函数,使其能够适应不同数量的待校准参数。 在参数校准过程中,经常会遇到需要调整…

    2025年12月14日
    000
  • 使用Python灵活配置不同数量变量的代码结构

    本文旨在提供一种灵活的代码结构,用于处理需要校准不同数量参数的情况。通过使用可变参数列表和参数索引,可以避免为每种参数组合编写重复的代码,从而提高代码的可维护性和可扩展性。本文将详细介绍如何实现这种结构,并提供示例代码和注意事项。 在参数校准过程中,经常会遇到需要校准的参数数量不确定的情况。例如,有…

    2025年12月14日
    000
  • Python处理学生成绩:从文件读取到统计分析

    本文档旨在提供一个Python程序,用于读取包含学生成绩信息的文本文件,计算综合成绩,并将结果写入新文件。同时,程序还将统计各分数段的学生人数,并计算班级平均分。通过本文,你将学习如何使用Python进行文件读写、数据处理和统计分析。 文件读取与数据解析 首先,我们需要从score1.txt文件中读…

    2025年12月14日
    000
  • Python处理学生成绩:从原始数据到统计分析的完整指南

    本文旨在提供一个全面的Python教程,指导读者如何读取包含学生学号、平时成绩和期末成绩的文本文件,计算总评成绩,并将结果写入新文件。同时,文章还将演示如何统计各分数段的学生人数,并计算班级平均分,旨在帮助读者掌握Python文件读写、数据处理和统计分析的基本技能。 1. 数据读取与解析 首先,我们…

    2025年12月14日
    000
  • Python处理学生成绩数据:计算总评、统计分数段及计算平均分

    本文旨在指导读者如何使用Python处理包含学生学号、平时成绩和期末成绩的文本文件,计算每个学生的总评成绩,并将结果写入新文件。同时,统计各分数段人数,并计算全班平均分。通过本文,读者将掌握文件读写、数据处理、循环控制和统计计算等常用Python编程技巧。 问题分析与改进 原始代码存在的主要问题是:…

    2025年12月14日
    000
  • 如何正确读取和处理文本文件中的数据并进行计算

    本文旨在帮助读者理解如何从文本文件中读取数据,进行数据处理和计算,并将结果写入新的文件。重点讲解了文本文件读取、数据分割、类型转换、字典创建、数据计算以及文件写入等关键步骤,并提供代码示例和注意事项,帮助读者避免常见的错误,从而实现高效的数据处理。 在处理文本文件中的数据时,经常需要读取文件内容,然…

    2025年12月14日
    000
  • Python如何连接MySQL数据库?PyMySQL详细使用教程

    %ignore_a_1%是python连接mysql数据库的首选工具。1.安装pymysql:使用pip install pymysql命令安装;2.连接数据库:通过pymysql.connect()方法建立连接,并使用cursor执行sql语句;3.使用连接池:通过dbutils.pooled_d…

    2025年12月14日 好文分享
    000
  • Python怎样操作SQLAlchemy?ORM高级用法

    要掌握sqlalchemy orm高级用法,关键在于查询优化、关系管理与结果处理。1. 使用selectinload和joinedload预加载关联数据,避免n+1查询问题;2. 通过defer延迟加载非必要字段,提升查询性能;3. 合理使用limit、offset与yield_per实现高效分页;…

    2025年12月14日 好文分享
    000
  • 将RGB颜色转换为最接近的ANSI控制台颜色

    本文旨在提供一个实用的教程,指导如何将任意RGB颜色值转换为控制台有限的ANSI颜色码。核心方法是利用欧几里得距离计算,在预定义的ANSI颜色调色板中找到与给定RGB颜色最接近的匹配项。这对于在终端中显示简化图像数据或进行颜色量化时非常有用,特别是在Python环境中。 1. 理解问题背景 在终端或…

    2025年12月14日
    000
  • Python如何实现物体检测?YOLO模型部署方案

    要在python中部署yolo进行物体检测,可按照以下步骤操作:1. 使用yolov5官方模型快速部署,通过pip安装依赖并运行detect.py脚本;2. 自定义模型加载与推理流程,使用torch.hub加载模型并手动调用推理函数;3. 部署为服务,利用flask创建rest api接收图片并返回…

    2025年12月14日 好文分享
    000
  • 如何用Python开发GUI程序?Tkinter控件详解

    tkinter的优势在于内置无需额外安装、跨平台支持良好、学习曲线平缓,适合快速开发小型工具;局限是界面风格较老旧,复杂ui和高性能图形渲染能力有限。1. 优势:内置标准库,跨平台运行,上手简单;2. 局限:默认界面不够现代化,复杂设计支持不足。常用控件包括label、button、entry、te…

    2025年12月14日 好文分享
    000

发表回复

登录后才能评论
关注微信