0

0

在Java中使用Gson向JSON文件追加新对象而非覆盖的教程

聖光之護

聖光之護

发布时间:2025-11-21 14:37:01

|

960人浏览过

|

来源于php中文网

原创

在java中使用gson向json文件追加新对象而非覆盖的教程

本教程旨在解决Java中向JSON文件追加新对象时,文件内容被覆盖的问题。我们将详细介绍如何利用GSON库实现“读取-解析-修改-写入”的流程,确保在不丢失现有数据的前提下,将新的JSON对象安全地添加到文件中,并提供将单个根对象转换为JSON数组以支持追加操作的策略。

引言:理解JSON文件追加的挑战

在Java应用程序中处理JSON数据时,一个常见的需求是将新的JSON对象添加到现有的JSON文件中,而不是完全覆盖其内容。例如,在一个员工管理系统中,每次添加新员工时,我们希望将新员工的JSON数据追加到 EmployeeData.json 文件中,而不是每次都创建一个只包含最新员工信息的新文件。

原始代码中,通常会直接将新的JSON字符串写入文件:

// 简化示例,与问题中的GSONFileWriter类似
public void writeJsonToFile(String jsonContent, String filePath) {
    try (PrintWriter writer = new PrintWriter(filePath, StandardCharsets.UTF_8)) {
        writer.write(jsonContent); // 此操作会覆盖文件原有内容
    } catch (IOException e) {
        e.printStackTrace();
    }
}

这种方法的问题在于,PrintWriter 或 FileWriter 在默认情况下会覆盖目标文件的所有内容。为了实现追加操作,我们需要采取一种“读取-解析-修改-写入”的策略。

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

核心策略:读取、解析、修改与写入

要向JSON文件追加数据,我们需要遵循以下步骤:

  1. 读取现有JSON文件内容: 将整个JSON文件作为字符串读取到内存中。
  2. 解析JSON字符串: 使用GSON库将读取到的JSON字符串解析成可操作的JSON结构(如 JsonElement, JsonObject 或 JsonArray)。
  3. 修改JSON结构: 将新的JSON对象添加到现有的JSON结构中。这一步至关重要,特别是当文件最初只包含一个JSON对象时,为了支持追加,需要将其转换为一个JSON数组。
  4. 写入修改后的JSON结构: 将更新后的JSON结构序列化回JSON字符串,并将其写回文件。

详细步骤与示例代码

我们将创建一个实用方法 appendObjectToJsonFile,它负责处理上述所有逻辑。

CA.LA
CA.LA

第一款时尚产品在线设计平台,服装设计系统

下载

1. 准备工作:引入GSON库和定义数据模型

首先,确保你的项目中已引入GSON库。如果你使用Maven,可以在 pom.xml 中添加以下依赖:


    com.google.code.gson
    gson
    2.10.1 

接下来,定义一个用于存储员工信息的数据模型(POJO):

import java.util.UUID;

// Employee.java
class Employee {
    String id;
    String name;
    int age;
    double salary;

    public Employee(String name, int age, double salary) {
        this.id = UUID.randomUUID().toString(); // 自动生成ID
        this.name = name;
        this.age = age;
        this.salary = salary;
    }

    // 默认构造函数或从JSON反序列化时可能需要
    public Employee() {}

    // Getter和Setter(可选,但通常推荐)
    public String getId() { return id; }
    public void setId(String id) { this.id = id; }
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
    public int getAge() { return age; }
    public void setAge(int age) { this.age = age; }
    public double getSalary() { return salary; }
    public void setSalary(double salary) { this.salary = salary; }

    @Override
    public String toString() {
        return "Employee{" +
               "id='" + id + '\'' +
               ", name='" + name + '\'' +
               ", age=" + age +
               ", salary=" + salary +
               '}';
    }
}

2. 实现 appendObjectToJsonFile 方法

这个方法将处理文件读取、JSON解析、结构修改和最终写入。

import com.google.gson.*;
import java.io.*;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Paths;

public class JsonFileAppender {

    // 使用GsonBuilder创建带美化打印的Gson实例,方便阅读
    private static final Gson GSON = new GsonBuilder().setPrettyPrinting().create();

    /**
     * 向指定的JSON文件追加一个新的对象。
     * 如果文件不存在、为空或内容为单个JSON对象,则将其转换为JSON数组并追加。
     * 如果文件已是JSON数组,则直接追加新对象。
     *
     * @param filePath 要追加的JSON文件路径
     * @param newObject 要追加的新Java对象
     * @throws IOException 如果发生文件读写错误
     * @throws JsonSyntaxException 如果文件内容不是有效的JSON
     */
    public static void appendObjectToJsonFile(String filePath, Object newObject) throws IOException {
        File jsonFile = new File(filePath);
        JsonArray jsonArray; // 最终要写入的JSON数组

        // 检查文件是否存在或是否为空
        if (!jsonFile.exists() || jsonFile.length() == 0) {
            // 文件不存在或为空,初始化一个新的JSON数组
            jsonArray = new JsonArray();
        } else {
            // 文件存在且有内容,读取并解析
            String fileContent = new String(Files.readAllBytes(Paths.get(filePath)), StandardCharsets.UTF_8);
            try {
                JsonElement rootElement = JsonParser.parseString(fileContent);

                if (rootElement.isJsonArray()) {
                    // 如果根元素已经是JSON数组,直接获取
                    jsonArray = rootElement.getAsJsonArray();
                } else if (rootElement.isJsonObject()) {
                    // 如果根元素是单个JSON对象,将其包装到一个新的JSON数组中
                    jsonArray = new JsonArray();
                    jsonArray.add(rootElement.getAsJsonObject());
                    System.out.println("检测到文件为单个JSON对象,已将其转换为JSON数组以支持追加。");
                } else {
                    // 处理其他非数组非对象的JSON类型(如原始值、null)
                    // 在这种情况下,我们选择创建一个新的数组并覆盖旧内容,或者抛出异常
                    System.err.println("警告: 现有JSON文件根元素既非数组也非对象。将创建新的JSON数组并覆盖。");
                    jsonArray = new JsonArray();
                }
            } catch (JsonSyntaxException e) {
                // 文件内容不是有效的JSON,视为需要重新开始,创建一个新的JSON数组
                System.err.println("警告: 现有文件内容不是有效的JSON。将创建新的JSON数组并覆盖。错误信息: " + e.getMessage());
                jsonArray = new JsonArray();
            }
        }

        // 将新的Java对象转换为JsonElement并添加到JSON数组中
        jsonArray.add(GSON.toJsonTree(newObject));

        // 将更新后的JSON数组写回文件
        try (Writer writer = new FileWriter(filePath, StandardCharsets.UTF_8)) {
            GSON.toJson(jsonArray, writer);
        }
        System.out.println("对象已成功追加到文件: " + filePath);
    }

    // 主方法用于演示
    public static void main(String[] args) {
        String fileName = "EmployeeData.json";

        // 清理旧文件,确保每次运行都是新开始(可选)
        new File(fileName).delete();
        System.out.println("已删除旧文件:" + fileName + " (如果存在)。");

        // 第一次追加:文件不存在,将创建包含一个对象的数组
        System.out.println("\n--- 第一次追加 ---");
        try {
            Employee emp1 = new Employee("Alice", 30, 60000.0);
            appendObjectToJsonFile(fileName, emp1);
            System.out.println("文件内容:\n" + new String(Files.readAllBytes(Paths.get(fileName)), StandardCharsets.UTF_8));
        } catch (IOException e) {
            e.printStackTrace();
        }

        // 第二次追加:文件已是数组,直接追加
        System.out.println("\n--- 第二次追加 ---");
        try {
            Employee emp2 = new Employee("Bob", 25, 55000.0);
            appendObjectToJsonFile(fileName, emp2);
            System.out.println("文件内容:\n" + new String(Files.readAllBytes(Paths.get(fileName)), StandardCharsets.UTF_8));
        } catch (IOException e) {
            e.printStackTrace();
        }

        // 第三次追加:文件仍是数组,继续追加
        System.out.println("\n--- 第三次追加 ---");
        try {
            Employee emp3 = new Employee("Charlie", 35, 70000.0);
            appendObjectToJsonFile(fileName, emp3);
            System.out.println("文件内容:\n" + new String(Files.readAllBytes(Paths.get(fileName)), StandardCharsets.UTF_8));
        } catch (IOException e) {
            e.printStackTrace();
        }

        // 演示如何读取所有员工
        System.out.println("\n--- 从文件中读取所有员工 ---");
        try {
            String fileContent = new String(Files.readAllBytes(Paths.get(fileName)), StandardCharsets.UTF_8);
            JsonElement rootElement = JsonParser.parseString(fileContent);
            if (rootElement.isJsonArray()) {
                JsonArray employeesArray = rootElement.getAsJsonArray();
                for (JsonElement element : employeesArray) {
                    Employee employee = GSON.fromJson(element, Employee.class);
                    System.out.println("读取到员工: " + employee);
                }
            } else {
                System.out.println("文件根元素不是JSON数组,无法按列表读取。");
            }
        } catch (IOException | JsonSyntaxException e) {
            e.printStackTrace();
        }
    }
}

代码执行结果示例

首次运行 main 方法后,EmployeeData.json 文件内容将类似如下:

[
  {
    "id": "d40507a7-a802-4494-9a0c-5a97a0a4d0bf",
    "name": "Alice",
    "age": 30,
    "salary": 60000.0
  },
  {
    "id": "e6f7g8h9-i0j1-k2l3-m4n5-o6p7q8r9s0t1",
    "name": "Bob",
    "age": 25,
    "salary": 55000.0
  },
  {
    "id": "f1g2h3i4-j5k6-l7m8-n9o0-p1q2r3s4t5u6",
    "name": "Charlie",
    "age": 35,
    "salary": 70000.0
  }
]

可以看到,每次追加操作都成功地将新员工对象添加到了JSON数组中,而不是覆盖整个文件。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

833

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

738

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

734

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

6

2026.01.16

热门下载

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

精品课程

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

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.8万人学习

Java 教程
Java 教程

共578课时 | 46.5万人学习

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

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