
本文讲解如何将数据库中存储的 json 格式物品列表(如角色背包数据)安全解码为 php 关联数组,并通过 foreach 循环逐项提取字段(如 name、count、info),避免因误用数组键名导致的错误。
你提供的 JSON 数据是一个索引数组(indexed array),其顶层结构是方括号 [] 包裹的多个对象,而非以 "name" 为键的关联数组。因此,直接写 $inventoryjson['name'] 会触发 PHP 警告(Notice: Undefined index: name),因为 $inventoryjson 实际上是一个数字索引数组(例如 $inventoryjson[0], $inventoryjson[1]…),而非键名为 'name' 的字典。
✅ 正确做法是:先用 json_decode($json, true) 将 JSON 字符串转为 PHP 关联数组(即“纯数组”模式),再用 foreach 遍历每一项:
$inventory = $karakter['inventory']; // 从数据库读取的 JSON 字符串
$items = json_decode($inventory, true);
// 检查解码是否成功(强烈建议添加错误处理)
if (json_last_error() !== JSON_ERROR_NONE) {
throw new RuntimeException('Invalid JSON in inventory: ' . json_last_error_msg());
}
// 安全遍历每个物品
foreach ($items as $item) {
// 提取基础字段
$name = $item['name'] ?? 'unknown';
$count = $item['count'] ?? 0;
$slot = $item['slot'] ?? null;
$type = $item['type'] ?? 'item';
// 处理可能为 null、数组或对象的 info 字段
$info = $item['info'] ?? [];
if (is_object($info)) {
$info = (array) $info; // 转为数组便于统一处理
}
// 示例:输出物品简要信息
echo "Slot {$slot}: {$name} × {$count} | Type: {$type}\n";
// 若需访问 info 内部字段(如 phone 的 telno),可进一步判断
if (is_array($info) && isset($info['telno'])) {
echo " → Phone: {$info['telno']} ({$info['isim']})\n";
}
if (is_array($info) && isset($info['uniqueId'])) {
echo " → Key ID: {$info['uniqueId']}\n";
}
}⚠️ 关键注意事项:
- ❌ 不要写 $inventoryjson['name'] —— 因为 $inventoryjson 是 array[0], array[1]…,不是 ['name' => ...];
- ✅ 始终校验 json_decode() 结果,使用 json_last_error() 防止无效 JSON 导致静默失败;
- ✅ 使用空合并运算符 ?? 安全访问可能缺失的字段(如某些 item 没有 info);
- ✅ 对 info 字段做类型判断(is_array() / is_object()),避免 foreach(null) 报错;
- ? 若 JSON 来自用户输入或不可信来源,务必进行内容校验(如白名单字段、数值范围限制等)。
通过以上方式,你就能稳定、清晰、可扩展地处理任意长度的 JSON 物品列表,无论是渲染前端界面、生成数据库记录,还是触发游戏逻辑,都具备良好的健壮性与可维护性。
立即学习“PHP免费学习笔记(深入)”;











