0

0

使用Gson将JSON对象正确反序列化为Java对象教程

心靈之曲

心靈之曲

发布时间:2025-10-05 13:36:02

|

673人浏览过

|

来源于php中文网

原创

使用gson将json对象正确反序列化为java对象教程

本文旨在解决使用Gson库将JSON字符串反序列化为Java对象时常见的IllegalStateException: Expected BEGIN_OBJECT but was STRING错误。通过分析错误的根源——JSON结构与Java对象映射的误解,并提供正确的反序列化方法,帮助开发者理解如何将整个JSON对象直接映射到对应的Java类,从而避免不必要的迭代和类型不匹配问题,确保数据转换的准确性与效率。

理解JSON到Java对象的映射挑战

在Java开发中,将JSON数据转换为对应的Java对象(即反序列化)是常见的操作。Gson是一个流行的Java库,用于实现这一功能。然而,如果不正确地理解JSON数据的结构与Java对象模型之间的对应关系,很容易遇到运行时错误,例如java.lang.IllegalStateException: Expected BEGIN_OBJECT but was STRING at path $。这个错误通常意味着Gson在尝试解析一个JSON对象时,却发现了一个字符串(或其他非对象类型),或者尝试将一个JSON的子元素当作整个对象进行解析。

考虑以下JSON数据结构:

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

为了将上述JSON数据映射到Java对象,我们首先需要定义匹配的Java类。根据JSON的层级结构,我们可以定义Person和Value两个类:

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;
import lombok.ToString;

@Getter
@Setter
@Data
@NoArgsConstructor
@AllArgsConstructor
@ToString
public class Person  {
    String type;
    String key;
    Value value; // 注意这里是Value类型,对应JSON中的"value"对象
}
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;
import lombok.ToString;
import java.util.Map;

@Data
@Setter
@NoArgsConstructor
@AllArgsConstructor
@Getter
@ToString
public class Value  {
    String name;
    Map car;    // 对应JSON中的"car"对象
    Map rocket; // 对应JSON中的"rocket"对象
}

错误的反序列化尝试及原因分析

假设我们已经通过JsonParser将JSON字符串解析为了JsonObject:

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

JSON.NET 简单的使用 中文WORD版
JSON.NET 简单的使用 中文WORD版

本文档主要讲述的是JSON.NET 简单的使用;JSON.NET使用来将.NET中的对象转换为JSON字符串(序列化),或者将JSON字符串转换为.NET中已有类型的对象(反序列化?)。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看

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

// 假设 input.readUTF() 提供了上述JSON字符串
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" +
        "}";
JsonObject jsonObject = JsonParser.parseString(jsonString).getAsJsonObject();
Gson gson = new Gson();

// 错误的尝试
// for (Map.Entry entry : jsonObject.entrySet()) {
//     Person person = gson.fromJson(entry.getValue(), Person.class); // 这里会抛出IllegalStateException
//     System.out.println(person);
// }

上述被注释掉的代码片段展示了一个常见的错误模式。开发者可能误以为需要遍历JsonObject的每个条目(entry),然后将每个条目的值反序列化为Person对象。然而,这与我们的JSON结构不符:

  1. JSON结构: 整个JSON对象本身就代表了一个Person对象,它的顶级键(type, key, value)直接对应Person类的字段。
  2. entry.getValue()的类型: 当遍历jsonObject.entrySet()时,entry.getValue()会返回不同类型的JsonElement:
    • 对于"type"和"key",entry.getValue()将是JsonPrimitive(代表JSON字符串"set"和"person")。
    • 对于"value",entry.getValue()将是JsonObject,但它代表的是Value类,而非Person类。
  3. 错误根源: 当gson.fromJson(entry.getValue(), Person.class)被调用,并且entry.getValue()是一个JsonPrimitive(例如字符串"set")时,Gson会期望它是一个完整的JSON对象(以{开头),但却得到了一个字符串。因此,它会抛出IllegalStateException: Expected BEGIN_OBJECT but was STRING。即使对于"value"对应的JsonObject,它也无法直接映射到Person.class,因为Value类缺少type和key字段。

正确的反序列化方法

解决这个问题的关键在于认识到:整个JSON对象(即jsonObject本身)就是我们要反序列化的Person对象。因此,我们不需要遍历它的内部条目,而是直接将整个JsonObject传递给gson.fromJson()方法。

import com.google.gson.Gson;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import java.io.StringReader;
import java.util.Map;

public class JsonToPersonConverter {

    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" +
                "}";

        // 1. 使用JsonParser解析JSON字符串为JsonObject
        JsonObject jsonObject = JsonParser.parseString(jsonString).getAsJsonObject();

        // 2. 创建Gson实例
        Gson gson = new Gson();

        // 3. 直接将整个JsonObject反序列化为Person对象
        Person person = gson.fromJson(jsonObject, Person.class);

        // 4. 打印结果验证
        System.out.println("成功反序列化为 Person 对象:");
        System.out.println(person);
        System.out.println("Person Type: " + person.getType());
        System.out.println("Person Key: " + person.getKey());
        System.out.println("Person Value Name: " + person.getValue().getName());
        System.out.println("Person Value Car Model: " + person.getValue().getCar().get("model"));
    }
}

输出结果:

成功反序列化为 Person 对象:
Person(type=set, key=person, value=Value(name=Elon Musk, car={model=Tesla Roadster, year=2018}, rocket={name=Falcon 9, launches=87}))
Person Type: set
Person Key: person
Person Value Name: Elon Musk
Person Value Car Model: Tesla Roadster

注意事项与最佳实践

  1. 匹配JSON结构与Java对象: 这是使用Gson进行反序列化的核心。确保你的Java类字段名与JSON键名一致(或使用@SerializedName注解进行映射),并且嵌套的JSON对象应对应嵌套的Java类。
  2. 直接反序列化: 如果整个JSON字符串或JsonObject代表一个完整的Java对象,就直接将其作为参数传递给gson.fromJson()方法。避免不必要的迭代或尝试将JSON的子元素反序列化为父对象。
  3. JsonParser与Gson的结合使用:
    • JsonParser用于将JSON字符串解析成通用的JsonElement(可以是JsonObject, JsonArray, JsonPrimitive或JsonNull)。当你需要检查JSON结构、动态处理不同类型的JSON数据,或者只提取JSON中的部分数据时,JsonParser非常有用。
    • Gson的fromJson()方法则负责将JsonElement或JSON字符串映射到具体的Java对象实例。
    • 在上述示例中,我们先用JsonParser获取JsonObject,再用Gson进行映射。实际上,如果直接从JSON字符串反序列化,也可以省略JsonParser这一步:Person person = gson.fromJson(jsonString, Person.class); 这样更为简洁。
  4. 错误信息解读: 当遇到IllegalStateException时,仔细阅读错误信息中的Expected BEGIN_OBJECT but was STRING at path $(或其他类型)。path $表示JSON的根部,path $.someKey表示someKey字段。这能帮助你快速定位JSON结构与Java对象模型不匹配的具体位置。
  5. Lombok的运用: 示例中使用了Lombok注解(如@Data, @Getter, @Setter, @NoArgsConstructor, @AllArgsConstructor, @ToString),它们能极大地简化Java Bean的编写,减少样板代码,使代码更简洁易读。

总结

正确使用Gson将JSON反序列化为Java对象,关键在于准确理解JSON数据的整体结构,并将其与Java对象模型进行一对一的映射。当整个JSON数据代表一个完整的Java对象时,应直接将该JSON数据(无论是字符串、Reader还是JsonObject)传递给gson.fromJson()方法,而不是错误地尝试迭代其内部元素。通过遵循这些原则,可以有效避免常见的IllegalStateException,确保数据转换的流畅和准确。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

419

2023.08.07

json是什么
json是什么

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

535

2023.08.23

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

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

311

2023.10.13

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

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

77

2025.09.10

string转int
string转int

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

463

2023.08.02

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

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

298

2023.08.03

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

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

212

2023.09.04

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

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

1501

2023.10.24

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

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

精品课程

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

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.6万人学习

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

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