
本教程详细阐述了python脚本如何高效且准确地向php传递包含多个json对象的数组。核心在于python端将所有独立的数据字典收集到一个列表中,然后将整个列表序列化为单一的json数组字符串输出。php端通过两次json_decode操作,首先解析外部的json数组结构,然后遍历数组,对每个元素(内部的json字符串)再次解码为关联数组,从而实现数据的便捷访问和处理。
1. 问题分析:Python多JSON输出与PHP解析困境
在Python脚本中,如果通过循环逐次打印独立的JSON字符串,例如:
{"key": "SWAT-107", "assignee": "Unassigned"}
{"key": "SWAT-98", "assignee": "Unassigned"}当PHP通过shell_exec捕获这些输出时,它会接收到一个由所有这些独立JSON字符串简单拼接而成的长字符串,例如:
"{\"key\": \"SWAT-107\", \"assignee\": \"Unassigned\"}{\"key\": \"SWAT-98\", \"assignee\": \"Unassigned\"}"这样的字符串并非一个合法的JSON数组或JSON对象。直接对它使用PHP的json_decode()函数将无法正确解析,因为它不符合JSON语法规范中数组或对象的起始和结束符。
即使Python尝试构建一个包含JSON字符串的列表,并将其作为字符串打印出来,如"['{\"key\": ...}', '{\"key\": ...}']",PHP接收到的仍然是一个字符串,而非可直接操作的数组。问题症结在于,PHP需要一个符合JSON数组标准的字符串,即以[开头,以]结尾,且内部元素之间用逗号分隔。
立即学习“PHP免费学习笔记(深入)”;
2. Python端的解决方案:构建并输出完整的JSON数组
解决此问题的关键在于Python脚本需要生成一个单一的、完整的JSON数组字符串,而不是多个独立的JSON字符串。这可以通过以下步骤实现:
- 收集数据字典: 在循环内部,将每个数据字典(例如issue_data)添加到Python列表(例如all_issues_data)中。
- 整体序列化: 在循环结束后,使用json.dumps()函数将整个列表all_issues_data序列化为一个JSON数组字符串。
- 打印输出: 将这个最终的JSON数组字符串打印到标准输出。
以下是修正后的Python代码示例:
import json
# 假设 Jira 类和相关方法已定义,这里使用模拟数据作为示例
# from your_jira_module import Jira
if __name__ == "__main__":
# jira = Jira() # 示例,实际使用时请取消注释并确保Jira类可用
all_issues_data = [] # 用于收集所有issue数据的列表
# 模拟从Jira获取数据并处理
# fields = jira.get_fields()
# jql_issues = jira.get_jql_search_issues(jql_search="project = SWAT AND resolution = Unresolved ORDER BY priority DESC, updated DESC")
# 模拟一些issue数据,结构与Jira issue对象类似
mock_issues = [
{'key': 'SWAT-107', 'fields': {'assignee': {'displayName': 'Unassigned'}}},
{'key': 'SWAT-98', 'fields': {'assignee': {'displayName': 'Unassigned'}}},
{'key': 'SWAT-100', 'fields': {'assignee': {'displayName': 'Unassigned'}}}
]
for issue in mock_issues: # 假设 issue 是一个字典或对象
issue_data = {}
issue_data['key'] = issue['key'] # 访问key
# 确保 assignee 字段存在且有 displayName 属性
if issue['fields']['assignee']:
issue_data['assignee'] = issue['fields']['assignee']['displayName']
else:
issue_data['assignee'] = 'Unassigned' # 处理未分配的情况
all_issues_data.append(issue_data) # 将处理后的字典添加到列表中
# 将整个列表序列化为JSON数组字符串并打印
print(json.dumps(all_issues_data))代码解释:
- all_issues_data = []:创建一个空列表,用于存储每个Jira issue处理后的字典。
- all_issues_data.append(issue_data):在循环中,每次处理一个issue,将其相关数据封装成一个字典issue_data,然后添加到all_issues_data列表中。
- print(json.dumps(all_issues_data)):循环结束后,all_issues_data列表包含了所有issue的字典数据。此时,使用json.dumps()函数将整个列表转换为一个标准的JSON数组字符串,例如 [{"key": "SWAT-107", "assignee": "Unassigned"}, {"key": "SWAT-98", "assignee": "Unassigned"}]。PHP将接收到这个完整且合法的JSON数组字符串。
3. PHP端的处理方法:双重JSON解码与关联数组
当Python脚本按照上述方法输出一个完整的JSON数组字符串后,PHP接收到的$output_json_string将是一个标准的JSON数组字符串。为了正确解析并遍历其中的数据,PHP需要进行两步解码:
- 第一次解码: 使用json_decode()解析由Python输出的外部JSON数组字符串,这将得到一个PHP数组。这个PHP数组的每个元素仍然是内部JSON对象的字符串表示。
- 第二次解码: 遍历这个PHP数组,对每个元素(即内部的JSON字符串)再次使用json_decode(),并传入true作为第二个参数,将其转换为PHP的关联数组,以便通过键名直接访问数据。
以下是修正后的PHP代码示例:
";
exit;
}
// 遍历PHP数组,对每个内部的JSON字符串进行第二次解码
if (is_array($decoded_output_array)) {
foreach ($decoded_output_array as $json_item_string) {
// 第二次解码:将内部JSON字符串解码为关联数组
// 'true' 参数确保解码为关联数组而非对象
$jira_data = json_decode($json_item_string, true);
// 检查第二次解码是否成功
if (json_last_error() !== JSON_ERROR_NONE) {
echo "PHP第二次JSON解码错误: " . json_last_error_msg() . " (Item: " . htmlspecialchars($json_item_string) . ")
";
continue; // 跳过当前错误项
}
// 现在可以像访问关联数组一样访问数据
if (is_array($jira_data) && isset($jira_data['key']) && isset($jira_data['assignee'])) {
echo "











