
本教程旨在解决python脚本向php返回多条json数据时遇到的常见问题。通过优化python脚本,将所有数据封装成一个完整的json数组输出,并在php端进行两阶段解码,即先解码整体json数组,再循环解码数组中的每个json字符串为关联数组,确保数据结构清晰、易于处理。
在开发Web应用时,我们经常需要利用Python处理复杂逻辑或数据,并通过PHP前端来调用并展示结果。一个常见的场景是Python脚本生成多条数据记录,每条记录以JSON格式表示,然后将其返回给PHP。然而,如果Python脚本只是简单地在循环中打印每个JSON对象,PHP接收到的将是一个由多个JSON字符串拼接而成的非标准字符串,导致PHP的json_decode函数无法正确解析。
原始的问题在于Python脚本在循环内部多次调用print(json.dumps(data)),导致每次迭代都输出一个独立的JSON字符串。当PHP通过shell_exec捕获这些输出时,它们被连接成一个单一的长字符串,例如:
{"key": "SWAT-107", "assignee": "Unassigned"} {"key": "SWAT-98", "assignee": "Unassigned"} ...这个字符串不是一个有效的JSON数组或对象,因此PHP的json_decode无法将其解析为一个可操作的数据结构。即使尝试将Python输出改为一个包含JSON字符串的Python列表,PHP接收到的仍然是一个字符串表示的列表,例如['{"key": ...}', '{"key": ...}'],这也不是一个标准的JSON格式。
解决此问题的关键在于确保Python脚本的最终输出是一个完整且标准的JSON结构,通常是一个JSON数组。这意味着我们需要在Python脚本中收集所有要返回的数据,然后一次性将整个数据结构序列化为JSON字符串并打印出来。
立即学习“PHP免费学习笔记(深入)”;
具体步骤如下:
以下是优化后的Python脚本示例:
import json
# 假设 Jira 类和 get_fields, get_jql_search_issues 方法已定义
# from PyJira.Jira import Jira
if __name__ == "__main__":
# jira = Jira() # 假设 Jira 实例已创建
output_data_list = [] # 用于存储所有数据字典的列表
# 模拟获取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")
# 模拟一些问题数据
jql_issues = [
type('obj', (object,), {'key': 'SWAT-107', 'fields': type('obj', (object,), {'assignee': type('obj', (object,), {'display_name': 'Unassigned'})})})(),
type('obj', (object,), {'key': 'SWAT-98', 'fields': type('obj', (object,), {'assignee': type('obj', (object,), {'display_name': 'Unassigned'})})})(),
type('obj', (object,), {'key': 'SWAT-100', 'fields': type('obj', (object,), {'assignee': type('obj', (object,), {'display_name': 'Admin'})})})(),
]
for issue in jql_issues:
data_item = {}
data_item['key'] = issue.key
# 假设 assignee 字段可能为空
assignee_name = issue.fields.assignee.display_name if issue.fields.assignee else "Unassigned"
data_item['assignee'] = assignee_name
output_data_list.append(data_item) # 将字典添加到列表
# 将整个列表序列化为JSON数组字符串并打印
print(json.dumps(output_data_list))
# exit() # 通常不需要在脚本末尾显式调用 exit()现在,Python脚本的输出将是一个标准的JSON数组,例如:
[{"key": "SWAT-107", "assignee": "Unassigned"}, {"key": "SWAT-98", "assignee": "Unassigned"}, {"key": "SWAT-100", "assignee": "Admin"}]当Python脚本按照上述方式优化后,PHP通过shell_exec接收到的将是一个有效的JSON数组字符串。PHP端的处理将分为两个阶段:
以下是PHP端的处理示例:
<?php
// 假设您的Python脚本路径和执行环境配置正确
$python_script_command = "cd .. && cd python/pyjira && pipenv run python PyJira/Jira.py";
$output_json_string = shell_exec($python_script_command);
// 确保 shell_exec 没有返回错误或空字符串
if ($output_json_string === null || $output_json_string === false) {
die("Error executing Python script or no output received.");
}
// 第一次解码:将整个JSON数组字符串解码为PHP数组
// 此时 $decoded_output 会是一个PHP数组,其中每个元素是PHP对象
$decoded_output = json_decode($output_json_string);
if (json_last_error() !== JSON_ERROR_NONE) {
die("JSON decode error (first stage): " . json_last_error_msg());
}
if (!is_array($decoded_output)) {
die("Python script did not return a valid JSON array.");
}
echo "<h2>Jira Issues:</h2>";
foreach ($decoded_output as $jira_item_object) {
// 第二次解码:将每个PHP对象(代表一个Jira问题)转换为关联数组
// 'true' 参数确保将对象转换为关联数组
$jira_data = (array) $jira_item_object; // 或者 json_decode(json_encode($jira_item_object), true); 更通用但稍慢
// 检查是否成功转换为数组且包含预期的键
if (is_array($jira_data) && isset($jira_data['key']) && isset($jira_data['assignee'])) {
echo "Key: " . htmlspecialchars($jira_data['key']) . " - Assignee: " . htmlspecialchars($jira_data['assignee']) . "<br>";
} else {
echo "Invalid item structure encountered.<br>";
}
}
?>代码说明:
通过遵循这些实践,您可以有效地在Python和PHP之间传递复杂的JSON数据结构,实现健壮可靠的跨语言通信。
以上就是Python与PHP交互:正确传递JSON数组的实践指南的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号