0

0

Gson在Java中将JSON转换为Java对象:常见错误与正确实践

聖光之護

聖光之護

发布时间:2025-10-05 10:46:02

|

985人浏览过

|

来源于php中文网

原创

Gson在Java中将JSON转换为Java对象:常见错误与正确实践

本文详细介绍了在Java中使用Gson库将JSON数据反序列化为Java对象的正确方法。通过一个具体的示例,我们探讨了常见的IllegalStateException错误原因,即错误地迭代JSON顶层元素并尝试将非对象类型转换为复杂对象。文章提供了修正后的代码示例,强调了Java对象结构与JSON数据结构精确匹配的重要性,并指导读者如何高效、无误地完成JSON到Java对象的转换。

1. Gson与JSON反序列化简介

在现代java应用开发中,json(javascript object notation)已成为数据交换的事实标准。将json数据转换为java对象(反序列化)是处理外部数据的重要环节。gson是google提供的一个功能强大的java库,用于在java对象和json数据之间进行序列化和反序列化。它以其简洁的api和灵活的配置广受欢迎。

Gson进行反序列化的核心方法是fromJson()。它通常接受一个JSON字符串或JsonElement对象,以及一个目标Java类的Class对象,然后返回一个该目标类的实例。

2. 示例JSON结构与Java对象定义

为了更好地理解JSON到Java对象的转换过程,我们首先定义一个典型的JSON数据结构及其对应的Java类。

示例JSON数据:

{
  "type":"set",
  "key":"person",
  "value":{
    "name":"Elon Musk",
    "car":{
      "model":"Tesla Roadster",
      "year":"2018"
    },
    "rocket":{
      "name":"Falcon 9",
      "launches":"87"
    }
  }
}

对应的Java对象定义:

立即学习Java免费学习笔记(深入)”;

为了匹配上述JSON结构,我们需要定义两个Java类:Person和Value。这里我们使用Lombok注解来简化POJO(Plain Old Java Object)的样板代码,如构造器、Getter/Setter等。

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.Map;

// Person.java:对应JSON的顶层结构
@Data // 包含 @Getter, @Setter, @ToString, @EqualsAndHashCode
@NoArgsConstructor
@AllArgsConstructor
public class Person {
    String type;
    String key;
    Value value; // 嵌套的Value对象,对应JSON中的"value"字段
}

// Value.java:对应JSON中嵌套的"value"对象
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Value {
    String name;
    Map car;    // 对应JSON中的"car"对象
    Map rocket; // 对应JSON中的"rocket"对象
}

可以看到,Person类直接映射了整个JSON的顶层结构,其中value字段是一个Value类型的对象,而Value类则进一步映射了JSON中value对象内部的结构。car和rocket字段由于其内部结构简单,被映射为Map类型。

3. 常见错误分析:IllegalStateException的原因

在进行JSON反序列化时,一个常见的错误是java.lang.IllegalStateException: Expected BEGIN_OBJECT but was STRING at path $。这个错误通常发生在Gson期望解析一个JSON对象(由{开始)但却遇到了一个字符串或其他非对象类型时。

让我们分析一个可能导致此错误的错误代码示例:

LALAL.AI
LALAL.AI

AI人声去除器和声乐提取工具

下载
import com.google.gson.Gson;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import java.util.Map;

public class IncorrectDeserializationExample {

    // ... Person 和 Value 类定义同上 ...

    public static void main(String[] args) {
        String jsonString = "{\n" +
                            "  \"type\":\"set\",\n" +
                            "  \"key\":\"person\",\n" +
                            "  \"value\":{\n" +
                            "    \"name\":\"Elon Musk\",\n" +
                            "    \"car\":{\n" +
                            "      \"model\":\"Tesla Roadster\",\n" +
                            "      \"year\":\"2018\"\n" +
                            "    },\n" +
                            "    \"rocket\":{\n" +
                            "      \"name\":\"Falcon 9\",\n" +
                            "      \"launches\":\"87\"\n" +
                            "    }\n" +
                            "  }\n" +
                            "}";

        Gson gson = new Gson();

        // 1. 将整个JSON字符串解析为一个JsonObject
        JsonObject jsonObject = JsonParser.parseString(jsonString).getAsJsonObject();

        System.out.println("--- 错误的反序列化尝试 ---");
        // 2. 错误:尝试迭代JsonObject的entrySet并反序列化每个值
        for (Map.Entry entry : jsonObject.entrySet()) {
            try {
                // 问题所在:entry.getValue() 在某些情况下不是一个JSON对象
                Person person = gson.fromJson(entry.getValue(), Person.class);
                System.out.println("成功反序列化为Person (意外): " + person);
            } catch (IllegalStateException e) {
                System.err.println("错误:无法将键 '" + entry.getKey() + "' 对应的值反序列化为 Person 对象。原因:" + e.getMessage());
            } catch (Exception e) {
                System.err.println("处理键 '" + entry.getKey() + "' 时发生其他错误: " + e.getMessage());
            }
        }
    }
}

错误原因分析:

上述代码中,JsonParser.parseString(jsonString).getAsJsonObject() 会将整个JSON字符串解析为一个JsonObject,这个jsonObject代表了我们JSON示例的顶层结构。

问题出在随后的for (Map.Entry entry : jsonObject.entrySet()) 循环中。这个循环会遍历顶层JsonObject的所有键值对

  • 当entry.getKey()为"type"时,entry.getValue()是一个JsonPrimitive(代表字符串"set")。
  • 当entry.getKey()为"key"时,entry.getValue()是一个JsonPrimitive(代表字符串"person")。
  • 当entry.getKey()为"value"时,entry.getValue()是一个JsonObject(代表嵌套的Value对象)。

Person类被定义为映射一个完整的JSON对象(以{开头),其中包含type、key和value三个字段。当gson.fromJson(entry.getValue(), Person.class)被调用时:

  1. 如果entry.getValue()是JsonPrimitive(如"set"或"person"),Gson会期望它是一个JSON对象(BEGIN_OBJECT),但实际得到的是一个字符串(STRING),因此抛出IllegalStateException。
  2. 即使entry.getValue()是一个JsonObject(对应键"value"),它也只代表了Value对象,而不是Person对象。尝试将其反序列化为Person.class也会失败,因为Value对象缺少type和key字段,并且其内部结构与Person类的根结构不匹配。

简而言之,错误在于试图将JSON的局部元素(如一个字符串或一个嵌套对象)反序列化为与其结构不匹配的顶级Java对象。

4. 正确实践:直接反序列化整个JSON对象

解决IllegalStateException的关键在于理解JSON结构与Java对象之间的映射关系,并确保将正确的JSON部分反序列化到对应的Java类。

在我们的示例中,Person类被设计用来映射整个顶层JSON结构。因此,我们应该直接将代表整个JSON的JsonObject(或者原始的JSON字符串)反序列化为Person对象。

正确代码示例:

import com.google.gson.Gson;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
// ... Person 和 Value 类定义同上 ...

public class CorrectDeserializationExample {

    // ... Person 和 Value 类定义同上 ...

    public static void main(String[] args) {
        String jsonString = "{\n" +
                            "  \"type\":\"set\",\n" +
                            "  \"key\":\"person\",\n" +
                            "  \"value\":{\n" +
                            "    \"name\":\"Elon Musk\",\n" +
                            "    \"car\":{\n" +
                            "      \"model\":\"Tesla Roadster\",\n" +
                            "      \"year\":\"2018\"\n" +
                            "    },\n" +
                            "    \"rocket\":{\n" +
                            "      \"name\":\"Falcon 9\",\n" +
                            "      \"launches\":\"87\"\n" +

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

420

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

536

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

312

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

483

2023.08.02

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1502

2023.10.24

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.4万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.6万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号