
当 `pandas.read_html` 无法从网页中提取表格时,通常是由于表格内容通过 javascript 动态加载。本教程将指导您如何利用浏览器开发者工具识别后台数据请求,并使用 `requests` 库模拟这些 xhr 请求,直接获取原始 json 数据。随后,我们将这些数据转换为 `pandas.dataframe`,从而有效解决动态网页数据抓取难题,实现高效、稳定的数据提取。
pandas.read_html 是一个极其便捷的工具,能够从静态 HTML 页面中快速识别并提取表格数据。它通过解析 HTML 结构中的 <table> 标签来工作。然而,在现代 Web 开发中,许多网页的内容,特别是表格数据,并非直接嵌入在初始加载的 HTML 中。相反,它们通常通过 JavaScript 在页面加载完成后异步地从服务器请求(例如,使用 AJAX 或 XHR 请求)并动态渲染到 DOM 中。
在这种情况下,当您尝试使用 pd.read_html 访问一个包含动态加载表格的 URL 时,它只会看到初始的静态 HTML 骨架,而不会等待 JavaScript 执行并填充表格内容。因此,read_html 返回一个空列表是预期的行为,因为它在初始 HTML 中找不到任何 <table> 标签。
例如,对于以下代码,如果目标网页的表格是动态加载的,dfs 将会是一个空列表:
import pandas as pd
# 尝试使用 read_html 提取数据
dfs = pd.read_html("https://anex.us/grades/?dept=ENGR&number=102")
print(dfs) # 预期输出:[]即使尝试使用 Selenium 等工具获取渲染后的 HTML,如果表格内容是通过独立的 API 请求获取并填充的,直接获取 outerHTML 也可能只得到一个空表格结构,因为数据本身并未被捕获。
立即学习“前端免费学习笔记(深入)”;
解决动态加载表格问题的关键在于绕过浏览器渲染过程,直接模拟页面在后台发出的数据请求。这通常涉及以下步骤:
这是最关键的一步。大多数现代浏览器都提供了强大的开发者工具,可以帮助我们监控网络活动。
通过分析示例网页,我们可以发现,当页面加载时,它向 https://anex.us/grades/getData/ 发送了一个 POST 请求,并带有 dept 和 number 参数。
一旦识别出数据源及其请求参数,我们就可以使用 Python 的 requests 库来模拟这个请求。
import requests
import pandas as pd
# 1. 确定数据接口 URL
url = 'https://anex.us/grades/getData/'
# 2. 构建请求载荷 (Payload)
# 这些参数是从浏览器开发者工具的 "Payload" 或 "Form Data" 中获取的
payload = {'dept': 'ENGR', 'number': '102'}
# 3. 设置请求头 (Headers)
# 模仿浏览器请求,特别是 User-Agent,可以避免某些网站的反爬虫机制
headers = {
'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64; rv:120.0) Gecko/20100101 Firefox/120.0'
}
# 4. 发送 POST 请求
# 使用 data 参数传递表单数据
response = requests.post(url, data=payload, headers=headers)
# 5. 检查响应状态码
if response.status_code == 200:
# 6. 解析 JSON 响应
# 大多数动态数据接口会返回 JSON 格式的数据
data = response.json()
# 7. 将 JSON 数据转换为 pandas DataFrame
# 根据 JSON 结构,找到包含实际数据列表的键
if 'classes' in data:
df = pd.DataFrame(data['classes'])
print(df)
else:
print("JSON 响应中未找到 'classes' 键。")
else:
print(f"请求失败,状态码: {response.status_code}")
print(response.text) # 打印响应内容以帮助调试运行上述代码,您将获得一个包含所需表格数据的 pandas.DataFrame:
dept number section A B C D F I S U Q X prof year semester gpa 0 ENGR 102 20 18 17 8 2 3 0 0 0 1 0 AMINI N 2018 FALL 2.9375 1 ENGR 102 21 18 31 15 4 1 0 0 0 0 0 KOOLA P 2018 FALL 2.88405797101449 2 ENGR 102 22 10 28 16 2 3 0 0 0 0 0 SHAW S 2018 FALL 2.67796610169492 3 ENGR 102 26 9 24 10 4 6 0 0 0 0 0 SUBRAMANIAN R 2018 FALL 2.49056603773585 4 ENGR 102 201 21 12 1 1 0 0 0 0 0 0 IJAZ M 2018 FALL 3.51428571428571 .. ... ... ... .. .. .. .. .. .. .. .. .. .. ... ... ... ... 486 ENGR 102 507 27 14 7 2 5 0 0 0 13 0 IJAZ M 2023 SPRING 3.01818181818182 487 ENGR 102 508 24 7 10 1 4 0 0 0 4 0 IJAZ M 2023 SPRING 3 488 ENGR 102 540 12 3 2 1 0 0 0 0 0 0 ALVARADO L 2023 SPRING 3.44444444444444 489 ENGR 102 550 1 1 6 1 2 0 0 0 0 0 KOOLA P 2023 SPRING 1.81818181818182 490 ENGR 102 551 1 1 7 2 3 0 0 0 3 0 VILLAREAL S 2023 SPRING 1.64285714285714 [491 rows x 17 columns]
当 pandas.read_html 无法满足动态网页的数据提取需求时,通过浏览器开发者工具定位并模拟底层的 XHR 数据请求,是获取结构化数据的有效且高效的方法。requests 库提供了强大的功能来构建和发送这些请求,配合 pandas 库,可以轻松地将获取到的 JSON 数据转换为可分析的 DataFrame。掌握这一技能,将大大扩展您从复杂网页中提取数据的能力。
以上就是从动态网页中高效提取表格数据:超越 pandas read_html 的方法的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号