java要防止重复序列化的问题JSON.toJSONString转义问题

要防止重复序列化的问题JSON.toJSONString(entity)

20241213 10:29

背景:

我在设计业务实现echart图标渲染,业务接口实时性查询耗时很长,为了提高系统可用性和用户体验,采用中间表的方案——即在中间表中存储大JSON。
但是在自测环节发现数据库中的json串格式有被多次序列化JSON.toJSONString(entity) 。疑惑了 所以就去看了下同事相关的业务接口 ,不得了 同事写的接口有的加了JSON.toJSONString序列化, 有的没加,简直就是坑啊。我怎么确定他有没有序列化啊,总不至于每个都判断一次吧…不管了,直接兜底处理吧,又称shi上堆shi。

他这种写法 直接导致,我在获取解析json的时候会有格式报错,因为无法确定到底被序列化了几次!

具体解决方案如下:

Q、

我要存储json字符串到mysql text字段中。代码使用JSON.toJSONString(entity) ,可能entity本身就是json字符串格式,为了防止两次toJSONString的情况 怎么处理。

在这里插入图片描述

A、

在存储 JSON 字符串到 MySQL 的 TEXT 字段时,确实需要防止重复序列化的问题。如果 entity 本身已经是 JSON 字符串,再次使用 JSON.toJSONString() 会导致转义符等问题。

解决方案

在调用 JSON.toJSONString() 之前,可以先检查 entity 是否已经是一个合法的 JSON 字符串。以下是两种常见方法:


方法 1:使用 JSONValidator 验证

JSONValidator 是 FastJSON 提供的工具,用于验证字符串是否是合法的 JSON 格式。

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.JSONValidator;

public class JsonStorageExample {
    public static String ensureJsonString(Object entity) {
        if (entity instanceof String) {
            String jsonString = (String) entity;
            // 验证是否是合法的 JSON 字符串
            if (JSONValidator.from(jsonString).validate()) {
                return jsonString; // 已是 JSON 字符串,无需再次序列化
            }
        }
        // 对非字符串类型或非合法 JSON 字符串的对象进行序列化
        return JSON.toJSONString(entity);
    }

    public static void main(String[] args) {
        // 示例
        Object entity1 = "{\"key\":\"value\"}"; // 已经是 JSON 字符串
        Object entity2 = new JSONObject(); // 非 JSON 字符串
        ((JSONObject) entity2).put("key", "value");

        System.out.println(ensureJsonString(entity1)); // 输出:{"key":"value"}
        System.out.println(ensureJsonString(entity2)); // 输出:{"key":"value"}
    }
}

方法 2:手动判断是否是 JSON 格式

如果不想引入 JSONValidator,可以通过捕获异常的方式简单判断:

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;

public class JsonStorageExample {
    public static String ensureJsonString(Object entity) {
        if (entity instanceof String) {
            String jsonString = (String) entity;
            try {
                // 尝试解析字符串为 JSON 对象
                JSON.parse(jsonString);
                return jsonString; // 如果解析成功,说明是合法 JSON
            } catch (Exception e) {
                // 如果解析失败,说明不是合法 JSON,继续序列化
            }
        }
        // 对非字符串类型或非合法 JSON 字符串的对象进行序列化
        return JSON.toJSONString(entity);
    }

    public static void main(String[] args) {
        Object entity1 = "{\"key\":\"value\"}"; // 已经是 JSON 字符串
        Object entity2 = new JSONObject(); // 非 JSON 字符串
        ((JSONObject) entity2).put("key", "value");

        System.out.println(ensureJsonString(entity1)); // 输出:{"key":"value"}
        System.out.println(ensureJsonString(entity2)); // 输出:{"key":"value"}
    }
}

优化点

  1. 尽量避免重复序列化:
    • 优先判断是否是 JSON 字符串,减少不必要的序列化操作。
  2. 使用更高效的存储方式:
    • 如果可能,考虑使用 MySQL 的 JSON 类型字段(需要 MySQL 5.7 或更高版本)。这样可以直接存储 JSON 对象并支持 JSON 查询功能。
  3. 日志记录:
    • 如果 entity 格式异常或解析失败,可以记录日志以方便排查问题。

希望对你有所帮助! 关注我为程序员职业生涯蓄能破局。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值