
本文旨在解决Python中解析API响应时,将JSON数据转换为字典后,在尝试遍历和提取特定键值对时常遇到的`TypeError: string indices must be integers, not 'str'`错误。通过深入分析字典迭代行为,本文将指导读者如何正确地从单层JSON字典中直接访问和提取所需数据,从而避免不必要的循环并提高代码效率。
在Python开发中,处理来自API的JSON响应是常见的任务。通常,我们会使用json.loads()方法将JSON字符串转换为Python字典,然后从中提取所需信息。然而,在这个过程中,如果不理解Python字典的迭代机制,很容易遇到TypeError。本教程将详细解释这一问题及其解决方案。
当我们通过requests库获取API响应时,其内容通常是JSON格式的字符串。为了在Python中方便地操作这些数据,我们需要将其解码并转换为Python数据结构。
import requests
import json
def fetch_user_data(username):
"""
从API获取用户数据并将其解码为Python字典。
"""
api_url = f"https://api.sleeper.app/v1/user/{username}"
response = requests.get(api_url)
# 检查请求是否成功
if response.status_code == 200:
# 解码API响应内容(通常为UTF-8)
decoded_content = response.content.decode("UTF-8")
# 将JSON字符串解析为Python字典
user_data = json.loads(decoded_content)
return user_data
else:
print(f"Error fetching data: {response.status_code}")
return None
# 示例调用
example_username = "zeustrl"
user_profile = fetch_user_data(example_username)
if user_profile:
print("Fetched user data structure:")
print(json.dumps(user_profile, indent=4)) # 美化输出上述代码演示了如何从API获取数据,并将其转换为一个Python字典。例如,user_profile可能包含以下结构(为清晰起见,已格式化):
立即学习“Python免费学习笔记(深入)”;
{
"verification": null,
"username": "zeustrl",
"user_id": "766368574179770368",
"token": null,
"summoner_region": null,
"summoner_name": null,
"solicitable": null,
"real_name": null,
"phone": null,
"pending": null,
"notifications": null,
"metadata": null,
"is_bot": false,
"email": null,
"display_name": "ZeusTRL",
"deleted": null,
"data_updated": null,
"currencies": null,
"created": null,
"cookies": null,
"avatar": "f64d0b7a8d0e6fbf0d7856185875d972"
}这是一个典型的单层字典结构,其中包含多个键值对。
许多初学者在尝试从字典中提取特定值时,会错误地尝试遍历整个字典,然后使用循环变量作为键来访问字典。
考虑以下错误的代码片段:
Easily find JSON paths within JSON objects using our intuitive Json Path Finder
30
# 假设 user_profile 已经是一个字典
user_ids_list = []
# 错误示范:试图遍历字典并用循环变量作为键再次索引
for i in user_profile:
# 这里的 i 是字典的键(字符串),例如 "username", "user_id"
# 因此 i["username"] 实际上是 "username"["username"]
# 导致 TypeError
# user_ids_list.append({'username': i["username"], 'user_id': i["user_id"]})
pass # 避免实际运行错误代码当执行for i in user_profile:时,变量i在每次迭代中获取的是user_profile字典的键(key),而不是值或键值对。例如,第一次迭代i可能是字符串"verification",第二次是"username",依此类推。
因此,当尝试i["username"]时,Python会尝试对字符串"username"进行索引操作,而字符串索引只能是整数(表示字符位置),不能是另一个字符串"username"。这正是TypeError: string indices must be integers, not 'str'错误产生的原因。
要验证这一点,可以在循环中打印i:
# for i in user_profile:
# print(f"Current 'i' value: {i}, type: {type(i)}")
# Output would be:
# Current 'i' value: verification, type: <class 'str'>
# Current 'i' value: username, type: <class 'str'>
# ...对于像user_profile这样表示单个实体(例如一个用户)的字典,如果已知要提取的键,最直接且高效的方法是直接通过键名访问字典。不需要进行迭代。
def get_user_id_and_username(user_data_dict):
"""
从用户数据字典中提取 'user_id' 和 'username'。
"""
if user_data_dict and "user_id" in user_data_dict and "username" in user_data_dict:
return {
"user_id": user_data_dict["user_id"],
"username": user_data_dict["username"]
}
return None
# 假设 user_profile 已经通过 fetch_user_data 获取
if user_profile:
user_info = get_user_id_and_username(user_profile)
if user_info:
print("\nExtracted user information:")
print(user_info)
# 如果需要将这些信息添加到列表中
user_ids_list = []
user_ids_list.append(user_info)
print("\nUser info list:")
print(user_ids_list)在这个修正后的方法中:
通过遵循这些原则,您可以更有效地解析JSON数据并避免常见的TypeError,从而编写出更健壮和可维护的Python代码。
以上就是Python中解析JSON字典的常见陷阱与解决方案的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号