
本教程详细介绍了如何使用python通过gbgb官方api,高效地抓取指定日期范围内特定赛狗赛道的比赛结果。文章涵盖了api请求参数的动态构建、日期范围的迭代生成、json数据解析、特定赛道数据的筛选以及健壮的错误处理机制,最终将抓取到的数据保存为json文件,旨在提供一个自动化且可扩展的数据获取方案。
在进行数据分析或建立自动化系统时,从网站获取大量结构化数据是一项常见任务。对于GBGB(Great British Greyhound Board)的赛狗比赛结果,手动通过网页或硬编码特定比赛ID的方式效率低下且难以扩展。GBGB提供了一个API接口,允许开发者通过结构化的HTTP请求获取数据。本教程将指导您如何利用Python及其requests库,结合API的参数特性,实现对指定日期范围和特定赛道比赛结果的自动化抓取。
GBGB提供了一个结果查询API,其基础URL为 https://api.gbgb.org.uk/api/results。此API支持通过查询参数来筛选数据,常见的参数包括:
通过动态修改这些参数,我们可以构建灵活的数据抓取逻辑。
手动输入每个日期的URL显然不可行。Python的datetime模块可以帮助我们轻松生成一个日期序列。为了覆盖一个月的每一天,我们需要一个双层循环:外层循环遍历月份,内层循环遍历该月份的每一天。
立即学习“Python免费学习笔记(深入)”;
from datetime import datetime, timedelta
import requests
import json
# 定义数据存储列表
json_list = []
# API基础URL和固定参数
base_url = "https://api.gbgb.org.uk/api/results"
params = {
'page': '1',
'itemsPerPage': '200', # 假设每页最多200条,可根据实际情况调整
'race_type': 'race'
}
# 定义抓取数据的年份和日期范围
year_to_scrape = 2023
start_month = 10
end_month = 12 # 包含结束月份
for month in range(start_month, end_month + 1):
# 计算当前月份的天数
# 简化处理,不考虑闰年,2月固定28天
if month == 2:
get_days_in_month = 28
elif month in [4, 6, 9, 11]:
get_days_in_month = 30
else:
get_days_in_month = 31
for day in range(1, get_days_in_month + 1):
# 格式化日期为 YYYY-MM-DD
# 使用 f-string 的 :02d 格式化保证月份和日期始终是两位数
params['date'] = f'{year_to_scrape}-{month:02d}-{day:02d}'
# ... 后续进行API请求注意事项:
生成日期参数后,下一步是使用requests库向API发送GET请求,并解析返回的JSON数据。
try:
# 使用requests发送GET请求
response = requests.get(base_url, params=params)
response.raise_for_status() # 检查HTTP请求是否成功 (200 OK)
# 将响应转换为Python字典
page_context_dict = response.json()
items = page_context_dict.get('items', []) # 获取'items'列表,若不存在则为空列表
# ... 后续进行数据筛选
except requests.exceptions.HTTPError as errh:
print(f"HTTP Error for date {params['date']}: {errh}")
except requests.exceptions.ConnectionError as errc:
print(f"Error Connecting for date {params['date']}: {errc}")
except requests.exceptions.Timeout as errt:
print(f"Timeout Error for date {params['date']}: {errt}")
except requests.exceptions.RequestException as err:
print(f"Oops: Something Else for date {params['date']}: {err}")
except json.JSONDecodeError as json_err:
print(f"JSON Decode Error for date {params['date']}: {json_err}")错误处理: 在进行网络请求时,务必包含错误处理机制。try-except块能够捕获各种潜在问题,如网络连接错误、HTTP状态码错误(如404、500)以及JSON解析错误,从而使程序更加健壮。response.raise_for_status()是一个便捷的方法,它会在HTTP请求返回非200状态码时抛出HTTPError异常。
API返回的数据通常包含多个赛道的信息。如果只需要特定赛道的数据,我们需要在解析JSON后进行筛选。API响应中的每个比赛结果对象通常包含一个trackName字段。
specific_track_items = []
desired_track = "Swindon" # 指定您想要抓取的赛道名称
for item in items:
# 检查'trackName'是否存在且与目标赛道匹配
if "trackName" in item and item["trackName"] == desired_track:
specific_track_items.append(item)
# 将筛选后的数据添加到总列表中
json_list.extend(specific_track_items)注意事项:
将上述所有部分整合,形成一个完整的Python脚本:
import requests
import json
from datetime import datetime, timedelta
import calendar # 用于更精确地获取月份天数
json_list = []
base_url = "https://api.gbgb.org.uk/api/results"
params = {
'page': '1',
'itemsPerPage': '200', # 可根据API限制和需求调整
'race_type': 'race'
}
# --- 配置抓取参数 ---
year_to_scrape = 2023
start_month = 10
end_month = 12 # 包含结束月份
desired_track = "Swindon" # 替换为您想要抓取的赛道名称
output_filename = 'gbgb_results_swindon.json' # 输出文件名
print(f"开始抓取 {year_to_scrape} 年 {start_month} 月到 {end_month} 月,赛道:{desired_track} 的数据...")
for month in range(start_month, end_month + 1):
# 使用calendar模块获取指定月份的天数,考虑闰年
num_days_in_month = calendar.monthrange(year_to_scrape, month)[1]
for day in range(1, num_days_in_month + 1):
# 格式化日期字符串
params['date'] = f'{year_to_scrape}-{month:02d}-{day:02d}'
print(f"正在请求数据:{params['date']}")
try:
response = requests.get(base_url, params=params)
response.raise_for_status()
page_context_dict = response.json()
items = page_context_dict.get('items', [])
specific_track_items = []
for item in items:
if "trackName" in item and item["trackName"] == desired_track:
specific_track_items.append(item)
if specific_track_items:
json_list.extend(specific_track_items)
print(f" - 成功抓取到 {len(specific_track_items)} 条 {desired_track} 赛道数据。")
else:
print(f" - {desired_track} 赛道在 {params['date']} 没有数据或数据为空。")
except requests.exceptions.HTTPError as errh:
print(f" - HTTP Error for date {params['date']}: {errh}")
except requests.exceptions.ConnectionError as errc:
print(f" - Error Connecting for date {params['date']}: {errc}")
except requests.exceptions.Timeout as errt:
print(f" - Timeout Error for date {params['date']}: {errt}")
except requests.exceptions.RequestException as err:
print(f" - Oops: Something Else for date {params['date']}: {err}")
except json.JSONDecodeError as json_err:
print(f" - JSON Decode Error for date {params['date']}: {json_err}")
# 将所有抓取到的数据写入JSON文件
if json_list:
with open(output_filename, 'w', encoding='utf-8') as f:
json.dump(json_list, f, ensure_ascii=False, indent=4)
print(f"\n数据抓取完成!共抓取到 {len(json_list)} 条数据,已保存至 {output_filename}")
else:
print("\n没有抓取到任何数据。请检查日期范围、赛道名称或API连接。")
通过本教程,您已经学会了如何利用Python和GBGB API高效地抓取特定日期范围和赛道的数据。这种方法相比手动抓取具有显著的效率和可扩展性优势。
进一步的改进和注意事项:
通过遵循这些指导原则,您可以构建一个稳定、高效的Python脚本,用于从GBGB API获取所需的赛狗比赛数据。
以上就是Python教程:使用API高效抓取GBGB赛狗结果数据(指定日期与赛道)的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号