0

0

使用 Python 和 Boto3 库高效统计 AWS S3 存储桶中特定文件

心靈之曲

心靈之曲

发布时间:2025-08-21 17:28:01

|

709人浏览过

|

来源于php中文网

原创

使用 python 和 boto3 库高效统计 aws s3 存储桶中特定文件

本教程详细介绍了如何使用 Python 和 Boto3 库高效地统计 AWS S3 存储桶中符合特定命名模式的文件数量。文章重点阐述了 boto3.resource 相较于 boto3.client 在处理 S3 对象列表和分页方面的优势,并提供了结合前缀过滤与客户端精确匹配的完整代码示例,帮助用户实现自动化文件校验与管理。

引言:S3 文件计数挑战

在 AWS S3(Simple Storage Service)中管理大量文件时,经常需要统计特定目录或符合特定命名规则的文件数量。例如,对于视频切片存储场景,可能需要验证每个视频质量版本(如 144p, 360p)下是否存在预期数量的切片文件(如 file_000.ts, file_001.ts 等)。手动检查这些文件既耗时又容易出错,因此编写自动化脚本变得尤为重要。

Python 的 boto3 库是与 AWS 服务交互的官方 SDK,提供了强大的功能来操作 S3。然而,在使用 boto3 进行文件计数时,尤其是在文件数量庞大且分散在多个“文件夹”中时,需要注意一些关键点,如分页处理和精确的文件名匹配。

传统 boto3.client 方法的局限性

在最初尝试使用 boto3.client().list_objects_v2 方法时,可能会遇到以下挑战:

  1. 手动分页处理:list_objects_v2 每次请求最多返回 1000 个对象。如果 S3 路径下的文件数量超过此限制,需要通过检查响应中的 IsTruncated 字段和使用 NextContinuationToken 参数来手动迭代所有页面,才能获取完整的对象列表。若未正确处理分页,将导致计数不完整。
  2. 不精确的过滤:list_objects_v2 的 Prefix 参数用于服务器端初步过滤,但它只能匹配对象键的开头。如果需要根据对象键的中间或结尾部分进行更复杂的匹配(例如,匹配所有以 file_ 开头且以 .ts 结尾的文件),则需要在客户端(即 Python 脚本中)对返回的对象列表进行二次过滤。原始尝试中 obj['Key'].startswith('file_000.ts') 过于具体,无法统计所有 file_*.ts 文件。
  3. S3 路径解析:S3 中的“文件夹”实际上是对象键的一部分。例如,s3://bucket/folder/subfolder/file.txt 中,folder/subfolder/ 是 file.txt 对象键的一部分。在构建 Prefix 时,需要确保其与 S3 对象的实际键结构匹配。

推荐方案:使用 boto3.resource 进行高效计数

boto3.resource 接口提供了更高级别的抽象,相较于 boto3.client,它更面向对象,并能自动处理许多底层细节,包括分页。这使得代码更简洁、更易读。

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

零沫AI工具导航
零沫AI工具导航

零沫AI工具导航-AI导航新标杆,探索全球实用AI工具

下载

1. boto3.resource 的优势

  • 自动分页:resource 对象(如 bucket.objects.filter())在迭代时会自动处理分页,无需手动管理 ContinuationToken。
  • 更 Pythonic 的接口:提供了更直观的方法和属性来访问 S3 桶和对象。

2. 基本计数示例

以下是一个使用 boto3.resource 统计 S3 桶中特定前缀下,符合 file_*.ts 模式文件数量的基本函数:

import boto3

def count_specific_files_in_s3(bucket_name, s3_prefix, file_name_starts_with='file_', file_name_ends_with='.ts'):
    """
    统计S3桶中指定前缀下,符合特定命名模式的文件数量。

    Args:
        bucket_name (str): S3 桶的名称。
        s3_prefix (str): 要搜索的S3前缀(即“文件夹”路径)。
                         例如:'Financial_Freedom_Course_Kannada/00_Course_Trailer_New_update/144p/'
        file_name_starts_with (str): 文件名应以此字符串开头。
        file_name_ends_with (str): 文件名应以此字符串结尾。

    Returns:
        int: 符合条件的文件数量。
    """
    s3 = boto3.resource('s3')
    bucket = s3.Bucket(bucket_name)
    actual_count = 0

    print(f"正在检查 S3 路径: s3://{bucket_name}/{s3_prefix}...")

    # 使用 Prefix 进行服务器端初步过滤
    # bucket.objects.filter() 会自动处理分页
    for obj in bucket.objects.filter(Prefix=s3_prefix):
        # 进一步在客户端过滤,确保符合命名约定且不是S3模拟的文件夹对象
        # S3中没有真正的文件夹,以'/'结尾的键通常被视为文件夹
        if obj.key.startswith(f'{s3_prefix}{file_name_starts_with}') and \
           obj.key.endswith(file_name_ends_with) and \
           not obj.key.endswith('/'): # 排除S3模拟的文件夹对象
            actual_count += 1
            # print(f"  找到匹配文件: {obj.key}") # 可用于调试
    return actual_count

# 示例用法 (请替换为您的实际桶名和前缀)
# fixed_bucket_name = 'your-s3-bucket-name'
# example_prefix = 'your-folder-path/sub-folder/'
# count = count_specific_files_in_s3(fixed_bucket_name, example_prefix)
# print(f"在 s3://{fixed_bucket_name}/{example_prefix} 路径下找到 {count} 个文件。")

3. 深入理解 Prefix 和 Key 过滤

  • Prefix 参数: 这是 S3 服务端提供的过滤机制。它能有效地减少从 S3 传输到本地的数据量,提高效率。当您指定 Prefix='path/to/folder/' 时,S3 只会返回键以 path/to/folder/ 开头的对象。
  • obj.key 客户端过滤: 尽管 Prefix 很有用,但它可能不足以满足所有复杂的过滤需求。例如,您可能需要匹配文件名的特定模式,或者排除某些类型的文件。这时,您需要在 Python 代码中对 obj.key 属性进行字符串操作(如 startswith(), endswith(), in 或正则表达式)来精确匹配所需的文件。在上述示例中,我们结合了 startswith(f'{s3_prefix}{file_name_starts_with}') 和 endswith(file_name_ends_with) 来实现精确匹配,并排除了以 / 结尾的“文件夹”对象。

整合到实际应用场景:批量校验 S3 视频切片

结合原始问题中从 CSV 读取 URL 并写入结果的需求,我们可以构建一个完整的脚本来自动化这一过程。

假设 ldt_ffw_course_videos_temp.csv 文件包含以下结构:

course_video_s3_url,course_video_ts_file_cnt
Financial_Freedom_Course_Kannada/00_Course_Trailer_New_update/144p/,28
Financial_Freedom_Course_Kannada/00_Course_Trailer_New_update/360p/,54
Financial_Freedom_Course_Kannada/00_Course_Trailer_New_update/720p/,34

其中 course_video_s3_url 是相对于 S3 桶根目录的路径,course_video_ts_file_cnt 是该路径下期望的文件数量。

import csv
import boto3
from urllib.parse import urlparse # 用于解析S3 URL,如果需要

# 辅助函数:解析 S3 URL 获取桶名和前缀 (如果输入CSV中是完整S3 URL)
def parse_s3_url(s3_url):
    """从完整的S3 URL中解析出桶名和前缀路径。"""
    parsed = urlparse(s3_url)
    if parsed.scheme != 's3':
        raise ValueError("URL 必须以 's3://' 开头。")
    bucket_name = parsed.netloc
    # path 包含前导斜杠,需要去除
    prefix = parsed.path.lstrip('/')
    return bucket_name, prefix

# 改进的计数函数 (与上面示例相同)
def count_specific_files_in_s3(bucket_name, s3_prefix, file_name_starts_with='file_', file_name_ends_with='.ts'):
    s3 = boto3.resource('s3')
    bucket = s3.Bucket(bucket_name)
    actual_count = 0

    for obj in bucket.objects.filter(Prefix=s3_prefix):
        if obj.key.startswith(f'{s3_prefix}{file_name_starts_with}') and \
           obj.key.endswith(file_name_ends_with) and \
           not obj.key.endswith('/'):
            actual_count += 1
    return actual_count

# 输入和输出 CSV 文件名
input_csv_file = 'ldt_ffw_course_videos_temp.csv'
output_csv_file = 'file_count_result.csv'

# 假设 S3 桶名是固定的。请替换为您的实际桶名。
# 如果您的CSV中包含完整的S3 URL (如 s3://bucket-name/path/...), 则可以使用 parse_s3_url 函数动态获取桶名。
fixed_bucket_name = 'coursevideotesting' # 替换为你的S3桶名

# 主处理逻辑
try:
    with open(input_csv_file, mode='r', encoding='utf-8') as infile, \
         open(output_csv_file, mode='w', newline='', encoding='utf-8') as outfile:

        reader = csv.DictReader(infile)

        # 验证输入CSV文件是否包含必要的列
        required_columns = ['course_video_s3_url', 'course_video_ts_file_cnt']
        if not all(col in reader.fieldnames for col in required_columns):
            raise ValueError(f"输入CSV文件必须包含以下列: {', '.join(required_columns)}")

        #

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

531

2023.06.20

正则表达式不包含
正则表达式不包含

正则表达式,又称规则表达式,,是一种文本模式,包括普通字符和特殊字符,是计算机科学的一个概念。正则表达式使用单个字符串来描述、匹配一系列匹配某个句法规则的字符串,通常被用来检索、替换那些符合某个模式的文本。php中文网给大家带来了有关正则表达式的相关教程以及文章,希望对大家能有所帮助。

258

2023.07.05

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

766

2023.07.05

java正则表达式匹配字符串
java正则表达式匹配字符串

在Java中,我们可以使用正则表达式来匹配字符串。本专题为大家带来java正则表达式匹配字符串的相关内容,帮助大家解决问题。

219

2023.08.11

正则表达式空格
正则表达式空格

正则表达式空格可以用“s”来表示,它是一个特殊的元字符,用于匹配任意空白字符,包括空格、制表符、换行符等。本专题为大家提供正则表达式相关的文章、下载、课程内容,供大家免费下载体验。

357

2023.08.31

Python爬虫获取数据的方法
Python爬虫获取数据的方法

Python爬虫可以通过请求库发送HTTP请求、解析库解析HTML、正则表达式提取数据,或使用数据抓取框架来获取数据。更多关于Python爬虫相关知识。详情阅读本专题下面的文章。php中文网欢迎大家前来学习。

293

2023.11.13

正则表达式空格如何表示
正则表达式空格如何表示

正则表达式空格可以用“s”来表示,它是一个特殊的元字符,用于匹配任意空白字符,包括空格、制表符、换行符等。想了解更多正则表达式空格怎么表示的内容,可以访问下面的文章。

245

2023.11.17

正则表达式中如何匹配数字
正则表达式中如何匹配数字

正则表达式中可以通过匹配单个数字、匹配多个数字、匹配固定长度的数字、匹配整数和小数、匹配负数和匹配科学计数法表示的数字的方法匹配数字。更多关于正则表达式的相关知识详情请看本专题下面的文章。php中文网欢迎大家前来学习。

547

2023.12.06

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

1

2026.03.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Django 教程
Django 教程

共28课时 | 5万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.9万人学习

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

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