
本文讲解如何将多个结构相似的 json 解析对象(每个含 data 数组)合并为一个标准对象,其 data 属性为所有子项扁平化后的单一数组,并保持 stdclass 格式。
在实际开发中(尤其是处理多用户 API 响应时),常会遇到类似场景:对每个认证用户调用接口并解析 JSON,得到一组形如 ["data" => [obj1]] 的对象,最终需统一聚合为一个顶层对象,其 data 字段是所有用户返回项的合并数组(而非嵌套数组)。你当前的代码问题在于:
- 错误地将整个 $single_data->data(即一个单元素数组)直接赋值给 $merged_data[$key],导致生成的是二维数组 [[obj1], [obj2]];
- 缺少最终封装——目标结构是一个 stdClass 对象,且其 data 属性必须是一维对象数组。
✅ 正确做法分三步:
提取每个 data 数组中的首项(或遍历全部子项)
若每个响应的 data 数组只含一个对象(如示例),直接取 $single_data->data[0];若可能含多个,则需 array_merge() 或循环追加。累积到一维结果数组
使用 [] 追加,而非键名索引赋值,避免保留原始索引层级。封装为标准对象
用 (object)['data' => $flattenedArray] 构建最终结构。
以下是完整、可直接运行的解决方案:
$data = [];
foreach ($this->get_authenticated_users() as $user) {
// ... 你的业务逻辑(获取 $json_data)...
$data[] = json_decode($json_data); // 得到 [obj1, obj2, ...],每个 obj 含 data=>[item]
}
// ✅ 正确合并:提取所有 data 数组中的对象,扁平化为一维数组
$flattened_items = [];
foreach ($data as $single_data) {
if (isset($single_data->data) && is_array($single_data->data)) {
foreach ($single_data->data as $item) {
$flattened_items[] = $item; // 逐个追加,确保扁平化
}
}
}
// ✅ 封装为标准 stdClass 对象,data 为合并后的一维数组
$result = (object)[
'data' => $flattened_items
];
var_dump($result);? 关键注意事项:
- 不要用 $merged_data[$key] = $single_data->data[0]:这仍会按原索引(0,1,...)写入,且未处理 data 多元素情况;
- 始终校验 isset($single_data->data) 和 is_array():避免 NULL 或非数组导致 Notice;
- 若需按时间倒序排列(如示例中期望的 "2022-02-23" 在前),可在合并后添加:
usort($flattened_items, fn($a, $b) => strcmp($b->timestamp, $a->timestamp));
最终输出将严格符合预期结构:一个 stdClass 对象,其 data 属性为包含全部 timestamp/id 对象的一维数组,便于后续 JSON 输出或前端消费。










