
本文介绍如何将多个结构相似的 json 解析对象(每个含 data 数组)合并为一个统一对象,其 data 属性为所有子项合并后的单一数组,并保持标准 stdclass 结构。
在实际开发中(如聚合多用户 API 响应),我们常遇到类似场景:对每个认证用户调用接口并解析 JSON,得到多个形如 {"data": [{...}, {...}]} 的对象。原始代码将每个解析结果存入 $data[],形成一个对象数组;但最终目标是合并所有 data 子数组为一个扁平化的 data 数组,并包裹在单个 stdClass 对象中——而非嵌套多层数组或保留键名索引。
问题根源在于:每个 $single_data->data 本身就是一个单元素数组(如 [0] => stdClass{timestamp,id}),而原逻辑 $merged_data[$key] = $single_data->data 直接将该数组作为值赋给新数组的键,导致结果为二维数组 [[obj],[obj]],而非期望的一维 data 数组 [obj, obj]。
✅ 正确做法分三步:
- 提取每个 data 数组的首元素(因示例中每个 data 仅含 1 个对象,即 $single_data->data[0]);
- 逐个追加到一个新数组(使用 [] 或 array_merge());
- 将结果封装进 stdClass 对象的 data 属性中。
以下是优化后的完整实现:
$data = [];
foreach ($this->get_authenticated_users() as $user) {
// ... 中间处理逻辑(获取 $json_data)...
$data[] = json_decode($json_data);
}
// 合并所有 data 子项为单一数组
$flattened_data = [];
foreach ($data as $single_data) {
// 确保 data 存在且为数组,取其中每个元素(支持单个或多个)
if (isset($single_data->data) && is_array($single_data->data)) {
foreach ($single_data->data as $item) {
$flattened_data[] = $item;
}
}
}
// 构建最终目标结构:stdClass{data: [obj, obj, ...]}
$result = new stdClass();
$result->data = $flattened_data;
var_dump($result);? 关键注意事项:
- 若每个源对象的 data 可能包含多个对象(不止 1 个),务必使用 foreach($single_data->data as $item) 遍历,而非硬编码 $single_data->data[0],以保证健壮性;
- 建议添加 isset() 和 is_array() 检查,避免因某次响应缺失 data 字段导致 Notice 错误;
- 如需按时间戳排序,可在合并后执行:
usort($flattened_data, function($a, $b) { return strtotime($b->timestamp) - strtotime($a->timestamp); });
最终输出即为所求结构:一个 stdClass 对象,其 data 属性是包含全部提取对象的一维数组,键名自动递增([0], [1], ...),语义清晰、便于后续 JSON 序列化或前端消费。










