如何用PHP从JSON中提取和访问数据?

这是一个一般的参考问题和回答,涵盖了许多永无止境的“如何访问JSON中的数据?”问题。这里将处理在PHP中解码JSON和访问结果的基本知识。

我有JSON:

{
"type": "donut",
"name": "Cake",
"toppings": [
{ "id": "5002", "type": "Glazed" },
{ "id": "5006", "type": "Chocolate with Sprinkles" },
{ "id": "5004", "type": "Maple" }
]
}

我如何解码这在PHP和访问结果数据?

432755 次浏览

介绍

首先你有一根弦。JSON不是数组、对象或数据结构。JSON是一个基于文本的序列化格式-所以一个奇特的字符串,但仍然只是一个字符串。在PHP中使用json_decode()解码它。

 $data = json_decode($json);

你可能会发现:

这些是可以用JSON编码的东西。或者更准确地说,这些是可以用JSON编码的东西的PHP版本。

他们没有什么特别的。它们不是“JSON对象”或“JSON数组”。你已经解码了JSON -你现在有基本的日常PHP类型

对象将是stdClass的实例,这是一个内置类,它只是一个普通的事情,在这里不重要。


访问对象属性

访问其中一个对象的属性,方法与访问其他对象(例如$object->property)的公共非静态属性相同。

$json = '
{
"type": "donut",
"name": "Cake"
}';


$yummy = json_decode($json);


echo $yummy->type; //donut

访问数组元素

访问其中一个数组的元素的方式与访问其他数组(例如$array[0])的方式相同。

$json = '
[
"Glazed",
"Chocolate with Sprinkles",
"Maple"
]';


$toppings = json_decode($json);


echo $toppings[1]; //Chocolate with Sprinkles

foreach遍历它。

foreach ($toppings as $topping) {
echo $topping, "\n";
}
< p >釉面
巧克力碎
枫< / p >

或者乱搞任何无数内置数组函数


访问嵌套项

对象的属性和数组的元素可能是更多的对象和/或数组——你可以像往常一样简单地继续访问它们的属性和成员,例如$object->array[0]->etc

$json = '
{
"type": "donut",
"name": "Cake",
"toppings": [
{ "id": "5002", "type": "Glazed" },
{ "id": "5006", "type": "Chocolate with Sprinkles" },
{ "id": "5004", "type": "Maple" }
]
}';


$yummy = json_decode($json);


echo $yummy->toppings[2]->id; //5004

true作为第二个参数传递给json_decode()

当你这样做的时候,你得到的不是对象,而是关联数组——用字符串作为键的数组。同样,您可以像往常一样访问其中的元素,例如$array['key']

$json = '
{
"type": "donut",
"name": "Cake",
"toppings": [
{ "id": "5002", "type": "Glazed" },
{ "id": "5006", "type": "Chocolate with Sprinkles" },
{ "id": "5004", "type": "Maple" }
]
}';


$yummy = json_decode($json, true);


echo $yummy['toppings'][2]['type']; //Maple

访问关联数组项

当将JSON 对象解码为关联PHP数组时,可以使用foreach (array_expression as $key => $value)语法迭代键和值

$json = '
{
"foo": "foo value",
"bar": "bar value",
"baz": "baz value"
}';


$assoc = json_decode($json, true);
foreach ($assoc as $key => $value) {
echo "The value of key '$key' is '$value'", PHP_EOL;
}

打印

键'foo'的值是'foo value'
键“bar”的值为“bar value”
关键字“baz”的值为“baz value”


不知道数据是如何结构的

阅读文档,了解JSON的来源。

看看JSON——这里你看到花括号{}期望一个对象,这里你看到方括号[]期望一个数组。

使用print_r()命中已解码的数据:

$json = '
{
"type": "donut",
"name": "Cake",
"toppings": [
{ "id": "5002", "type": "Glazed" },
{ "id": "5006", "type": "Chocolate with Sprinkles" },
{ "id": "5004", "type": "Maple" }
]
}';


$yummy = json_decode($json);


print_r($yummy);

并检查输出:

stdClass Object
(
[type] => donut
[name] => Cake
[toppings] => Array
(
[0] => stdClass Object
(
[id] => 5002
[type] => Glazed
)


[1] => stdClass Object
(
[id] => 5006
[type] => Chocolate with Sprinkles
)


[2] => stdClass Object
(
[id] => 5004
[type] => Maple
)


)


)

它会告诉你在哪里有对象,在哪里有数组,以及它们成员的名称和值。

如果你在迷失之前只能深入到这里,那么就用print_r()击中:

print_r($yummy->toppings[0]);
stdClass Object
(
[id] => 5002
[type] => Glazed
)

这个方便的交互式JSON资源管理器中看看它。

把问题分解成更容易理解的小块。


json_decode()返回null

发生这种情况是因为:

  1. JSON完全由null组成。
  2. JSON是无效的-检查json_last_error_msg的结果或将其放入类似JSONLint的地方。
  3. 它包含嵌套深度超过512层的元素。这个默认的最大深度可以通过将一个整数作为第三个参数传递给json_decode()来覆盖。

如果你需要改变最大深度,你可能解决了错误的问题。弄清楚为什么你会得到如此深嵌套的数据(例如,你正在查询的生成JSON的服务有一个错误),并让这种情况不会发生。


对象属性名包含一个特殊字符

有时你会有一个对象属性名,它包含像连字符-或@ @这样的东西,不能用于文字标识符。相反,您可以使用花括号内的字符串文字来解决它。

$json = '{"@attributes":{"answer":42}}';
$thing = json_decode($json);


echo $thing->{'@attributes'}->answer; //42

如果你有一个整数属性,请参考:如何访问对象属性的名称,如整数?作为引用。


有人把JSON放到了你的JSON中

这很荒谬,但它确实发生了——JSON中有编码为字符串的JSON。解码,像往常一样访问字符串,解码,最终得到你需要的东西。

$json = '
{
"type": "donut",
"name": "Cake",
"toppings": "[{ \"type\": \"Glazed\" }, { \"type\": \"Maple\" }]"
}';


$yummy = json_decode($json);
$toppings = json_decode($yummy->toppings);


echo $toppings[0]->type; //Glazed

数据不适合内存

如果你的JSON对于json_decode()来说太大了,事情就开始变得棘手了。看到的:


如何排序

看到:参考:PHP中排序数组和数据的所有基本方法