当前位置:首页 > 文章列表 > 文章 > php教程 > PHP解析JSON数组的几种方法

PHP解析JSON数组的几种方法

2025-09-04 08:30:40 0浏览 收藏

知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个文章开发实战,手把手教大家学习《PHP解析JSON数组的实用方法》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!

PHP中如何解析和读取JSON中的数组数据

本教程详细阐述了在PHP中解析和读取JSON字符串中包含的数组数据的方法。文章首先介绍了使用json_decode函数将JSON字符串转换为PHP数组的基本操作,随后深入讲解了如何通过多层索引访问嵌套数组中的特定数据,并提供了迭代遍历嵌套数组的示例代码。此外,教程还引入了RecursiveArrayIterator等高级迭代器,以应对更复杂的JSON结构,并强调了JSON数据有效性检查及错误处理的重要性。

JSON(JavaScript Object Notation)作为一种轻量级的数据交换格式,在Web应用中被广泛用于前后端数据传输。在PHP中处理JSON数据是日常开发中的常见任务,尤其是当JSON结构中包含多层嵌套的数组时,如何高效、准确地读取这些数据成为了关键。

1. JSON数据结构概述

JSON数据主要由两种结构组成:

  • 对象(Object):由键值对组成,键是字符串,值可以是字符串、数字、布尔值、null、对象或数组。在PHP中通常被解析为关联数组。
  • 数组(Array):由有序的值列表组成,值可以是任意JSON数据类型。在PHP中通常被解析为索引数组。

当JSON中包含嵌套的对象和数组时,理解其层级关系对于正确解析至关重要。例如,以下JSON片段展示了典型的嵌套结构:

{
  "success": 1,
  "results": [
    {
      "Id": 202322021,
      "SportName": "Football",
      "HomeTeam": "Kayserispor",
      "optionMarkets": [
        {
          "name": {
            "value": "Match Result"
          },
          "options": [
            {
              "name": {
                "value": "Kayserispor"
              },
              "price": {
                "odds": 1.055
              }
            },
            {
              "name": {
                "value": "X"
              },
              "price": {
                "odds": 11
              }
            }
          ]
        },
        {
          "name": {
            "value": "Double Chance"
          },
          "options": [
            {
              "value": "Kayserispor or X"
            },
            {
              "value": "X or Artvin Hopaspor"
            }
          ]
        }
      ]
    }
  ]
}

在这个结构中,results是一个数组,其每个元素都是一个对象。每个对象内部又包含optionMarkets数组,而optionMarkets数组的每个元素又是一个对象,其中包含options数组,等等。

2. 使用 json_decode 解析JSON字符串

在PHP中,json_decode()函数是解析JSON字符串的核心工具。它将JSON格式的字符串转换为PHP变量。

函数签名:mixed json_decode ( string $json [, bool $associative = false [, int $depth = 512 [, int $options = 0 ]]] )

  • $json:要解码的JSON字符串。
  • $associative:如果设置为true,JSON对象将被转换为PHP关联数组;如果设置为false(默认),JSON对象将被转换为PHP stdClass对象。对于处理嵌套结构,通常建议设置为true以便于数组方式访问。

示例代码:

<?php
// 示例JSON字符串
$json_string = '{"success":1,"pager":{"page":1,"per_page":250,"total":1813},"results":[{"Id":202322021,"SportId":4,"SportName":"Football","RegionId":31,"RegionName":"Turkey","LeagueId":102729,"LeagueName":"Turkiye Kupasi","BetRadarId":29664196,"our_event_id":4157451,"IsPreMatch":true,"Date":"2021-10-27T17:45:00Z","updated_at":"1635354806","HomeTeam":"Kayserispor","HomeTeamId":230391,"AwayTeam":"Artvin Hopaspor","AwayTeamId":228914,"Markets":[],"optionMarkets":[{"properties":[],"comboPrevention":"NoFixtureCombo","templateCategory":{"category":"Gridable","id":0},"status":"Visible","isMain":true,"grouping":{"gridGroups":["ls8o6ll2v"],"detailed":[{"name":"Match Result","marketTabId":1,"marketHelpPath":"Football\/Match Result (Regular Time)","index":7930,"displayType":"Regular","group":0}],"parameters":{"marketType":"ThreeWay","period":"RegularTime","happening":"Goal"}},"id":29622716,"name":{"value":"Match Result","sign":"9Vlc5w=="},"minCombo":1,"options":[{"name":{"value":"Kayserispor","sign":"tyg4wg=="},"id":96896088,"sourceName":{"value":"1"},"status":"Visible","price":{"id":145532907,"odds":1.055,"americanOdds":-2000,"denominator":18,"numerator":1}},{"price":{"id":145532911,"odds":11,"americanOdds":1000,"numerator":10,"denominator":1},"status":"Visible","name":{"value":"X","sign":"JHhicw=="},"id":96896089},{"price":{"id":145532915,"odds":31,"americanOdds":3000,"numerator":30,"denominator":1},"status":"Visible","sourceName":{"value":"2"},"id":96896090,"name":{"value":"Artvin Hopaspor","sign":"2ruL4g=="}}],"parameters":[{"key":"Happening","value":"Goal","type":"String"},{"key":"MarketType","value":"3way","type":"String"},{"type":"String","value":"RegularTime","key":"Period"}]},{"minCombo":1,"id":29622721,"name":{"value":"Double Chance","sign":"WRDxhg=="},"parameters":[{"key":"Happening","type":"String","value":"Goal"},{"key":"MarketType","value":"DoubleChance","type":"String"},{"type":"String","value":"RegularTime","key":"Period"}],"options":[{"status":"Suspended","price":{"odds":1,"id":145532955},"id":96896100,"name":{"sign":"YrFHYg==","value":"Kayserispor or X"}},{"name":{"value":"X or Artvin Hopaspor","sign":"knPxTA=="},"id":96896101,"status":"Visible","price":{"americanOdds":825,"denominator":4,"numerator":33,"id":145532959,"odds":9.25}},{"price":{"id":145532963,"odds":1.04,"americanOdds":-2500,"numerator":1,"denominator":25},"status":"Visible","name":{"sign":"fwDTwg==","value":"Kayserispor or Artvin Hopaspor"},"id":96896102}],"comboPrevention":"NoFixtureCombo","properties":[],"isMain":false,"status":"Visible","templateCategory":{"id":0,"category":"Gridable"},"grouping":{"parameters":{"period":"RegularTime","marketType":"DoubleChance","happening":"Goal"},"detailed":[{"marketTabId":1,"name":"Double Chance","group":0,"displayType":"Regular","marketHelpPath":"Football\/Double Chance","index":17230}],"gridGroups":["e8vi2hdwz"]}},{"options":[{"price":{"numerator":87,"denominator":100,"americanOdds":-115,"odds":1.87,"id":155223689},"status":"Visible","name":{"sign":"5Rageg==","value":"Over 3,5"},"id":100328998},{"price":{"numerator":5,"denominator":6,"americanOdds":-120,"odds":1.83,"id":155223693},"status":"Visible","id":100328999,"name":{"value":"Under 3,5","sign":"KP8xUw=="}}],"parameters":[{"type":"Decimal","value":"3.5000","key":"DecimalValue"},{"key":"Happening","value":"Goal","type":"String"},{"type":"String","value":"Over\/Under","key":"MarketType"},{"key":"Period","type":"String","value":"RegularTime"}],"name":{"value":"Total Goals","sign":"Y9HR2w=="},"id":30811485,"minCombo":1,"spread":0.04,"grouping":{"parameters":{"period":"RegularTime","marketType":"OverUnder","attr":"3.5000","happening":"Goal","attrType":"Decimal"},"detailed":[{"name":"Over\/Under Total Goals","marketTabId":1,"subIndex":4,"displayType":"OverUnder","index":27330,"group":0},{"subIndex":4,"marketTabId":1,"group":0,"displayType":"OverUnder","index":35130},{"group":2,"displayType":"OverUnder","index":11730,"subIndex":4,"marketTabId":1,"name":"Over\/Under Total Goals"},{"index":19530,"displayType":"OverUnder","group":2,"marketTabId":1,"subIndex":4}],"gridGroups":["tabpo1c9n"]},"balanced":1,"status":"Visible","attr":"3,5","isMain":false,"templateCategory":{"category":"Gridable","id":0},"properties":[],"comboPrevention":"NoFixtureCombo"}]}]}';

$json_data = json_decode($json_string, true);

// 检查解码是否成功
if (json_last_error() !== JSON_ERROR_NONE) {
    echo "JSON解码错误: " . json_last_error_msg() . "\n";
    exit;
}

// 访问顶层数据
echo "Success: " . $json_data['success'] . "\n";
echo "Total Results: " . $json_data['pager']['total'] . "\n";
?>

3. 遍历和访问嵌套数组

一旦JSON字符串被解码为PHP关联数组,就可以使用标准的数组访问语法([])和循环结构(foreach)来遍历和提取嵌套数据。

以前面的JSON数据为例,我们将演示如何读取results数组中的每个比赛信息,并进一步深入读取每个比赛的optionMarkets及其options。

<?php
// ... (接上文的 $json_string 和 $json_data 定义) ...

if (!empty($json_data['results'])) {
    foreach ($json_data['results'] as $event) {
        // 访问直接属性
        echo "----------------------------------------\n";
        echo "比赛ID: " . $event['Id'] . "\n";
        echo "运动类型: " . $event['SportName'] . "\n";
        echo "主队: " . $event['HomeTeam'] . "\n";
        echo "客队: " . $event['AwayTeam'] . "\n";
        echo "联赛名称: " . $event['LeagueName'] . "\n";
        echo "日期: " . $event['Date'] . "\n";

        // 访问 optionMarkets 数组
        if (!empty($event['optionMarkets'])) {
            echo "  市场选项 (optionMarkets):\n";
            foreach ($event['optionMarkets'] as $market) {
                // 访问市场名称
                $market_name = isset($market['name']['value']) ? $market['name']['value'] : 'N/A';
                echo "    - 市场名称: " . $market_name . "\n";

                // 访问 options 数组
                if (!empty($market['options'])) {
                    echo "      选项 (options):\n";
                    foreach ($market['options'] as $option) {
                        $option_name = isset($option['name']['value']) ? $option['name']['value'] : 'N/A';
                        $option_price = isset($option['price']['odds']) ? $option['price']['odds'] : 'N/A';
                        echo "        -> 名称: " . $option_name . ", 赔率: " . $option_price . "\n";
                    }
                } else {
                    echo "      无可用选项。\n";
                }
            }
        } else {
            echo "  无可用市场选项。\n";
        }
    }
} else {
    echo "JSON数据中没有 'results' 数组或其为空。\n";
}
?>

代码解析:

  1. 首先通过$json_data['results']访问到顶层的results数组。
  2. 使用第一个foreach循环遍历results数组中的每一个比赛事件($event)。
  3. 在每个$event内部,直接访问HomeTeam、AwayTeam等简单键值。
  4. 为了访问optionMarkets这个嵌套数组,我们再次使用if (!empty(...))进行存在性检查,然后使用第二个foreach循环遍历optionMarkets中的每个市场($market)。
  5. 同理,对于options数组,也进行存在性检查,并使用第三个foreach循环遍历每个选项($option)。
  6. 在访问深层嵌套的值(如$market['name']['value']和$option['price']['odds'])时,务必进行isset()或empty()检查,以避免在某些键不存在时产生PHP警告或错误。

4. 使用 RecursiveArrayIterator 进行深度遍历

对于结构深度不确定或需要通用遍历所有键值对的场景,PHP的RecursiveArrayIterator和RecursiveIteratorIterator提供了一种更优雅的解决方案。它们允许你以递归方式遍历多维数组,而无需手动编写多层嵌套循环。

<?php
// ... (接上文的 $json_string 和 $json_data 定义) ...

echo "\n--- 使用 RecursiveArrayIterator 遍历所有键值对 ---\n";

// 将解码后的数据放入一个ArrayObject,以便RecursiveArrayIterator使用
$data_obj = new ArrayObject($json_data);

// 创建RecursiveIteratorIterator实例,它将遍历RecursiveArrayIterator生成的所有元素
$iterator = new RecursiveIteratorIterator(
    new RecursiveArrayIterator($data_obj)
);

foreach ($iterator as $key => $value) {
    // 获取当前元素的深度,用于格式化输出
    $depth = $iterator->getDepth();
    echo str_repeat("  ", $depth) . $key . ": " . (is_array($value) ? "Array" : $value) . "\n";
}
?>

代码解析:

  1. new ArrayObject($json_data):将PHP数组包装成一个ArrayObject,这是RecursiveArrayIterator的构造函数所接受的类型。
  2. new RecursiveArrayIterator($data_obj):创建一个递归数组迭代器,它知道如何深入遍历数组的子元素。
  3. new RecursiveIteratorIterator(...):这是一个外部迭代器,它接受一个递归迭代器作为参数,并将其扁平化,使其可以像单层数组一样被foreach遍历。getDepth()方法可以获取当前元素的嵌套深度。
  4. 这种方法适用于需要检查JSON中所有键值对的场景,但如果只需要特定路径下的数据,手动遍历可能更直接。

5. 注意事项与最佳实践

  • JSON有效性验证:在json_decode之前或之后,始终使用json_last_error()和json_last_error_msg()检查JSON字符串是否有效以及解码过程中是否发生错误。无效的JSON字符串会导致json_decode返回null。
  • 键的存在性检查:在访问数组元素之前,使用isset()或array_key_exists()检查键是否存在,尤其是在处理来自外部源的JSON数据时,以防止因键缺失而导致的PHP通知或错误。
  • 性能考量:对于非常大的JSON字符串(几十MB甚至更大),json_decode会一次性将整个字符串加载到内存中。如果内存成为瓶颈,可能需要考虑使用流式解析器(如json_stream库)来逐块处理JSON数据。
  • 数据类型转换:json_decode会尝试将JSON值转换为相应的PHP类型(例如,JSON数字转换为PHP整数或浮点数,JSON布尔值转换为PHP布尔值)。了解这些转换有助于避免意外的数据类型问题。
  • 编码问题:确保JSON字符串的编码是UTF-8,并且PHP脚本的编码设置也正确,以避免中文或其他非ASCII字符乱码。

总结

在PHP中读取JSON中的数组数据,核心在于理解JSON的嵌套结构,并结合json_decode()函数将其转换为PHP关联数组。通过多层foreach循环可以精确地访问特定路径下的数据,而RecursiveArrayIterator则为深度或通用遍历提供了强大的工具。在实际开发中,务必结合错误处理和键存在性检查,以确保代码的健壮性和稳定性。掌握这些技巧,将使你能够高效地处理各种复杂的JSON数据。

以上就是《PHP解析JSON数组的几种方法》的详细内容,更多关于的资料请关注golang学习网公众号!

GolangTCP连接池实现与优化技巧GolangTCP连接池实现与优化技巧
上一篇
GolangTCP连接池实现与优化技巧
Logstash日志收集配置详解与优化
下一篇
Logstash日志收集配置详解与优化
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    512次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    499次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    854次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    809次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    839次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    859次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    834次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码