Java中Redis存储复杂json格式数据并返回给前端

本文探讨了在JavaWeb项目中,利用Redis高效存储和读取复杂JSON数据的三种方法:直接字符串存储、对象序列化存储及使用hash存储。通过实例展示了如何针对复杂的JSON格式数据,采用hash方式分割存储,提高数据处理效率。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题背景

在Java Web项目中,经常需要前端请求数据,后台从数据库中查询并计算最后返回json格式数据给前端。而每次请求都需要计算一次可能比较浪费时间,这时我们可以将计算好的结果保存在redis中,下次请求时先判断redis中是否已经存在,如果是则直接从redis里取出返回,因为是在内存中,所以比较快。而自己在项目中遇到的json格式数据比较复杂,下面记录一下redis存储对象和json格式数据的几种方式以及遇到的问题。

存储方式

1. 直接使用String存储

String类型是Redis中最简单的类型了,每个key对应一个String,我们可以直接将要存储的对象转换成json字符串,代码如下:

	//存储
	public static void setJsonString(String key, Object obj) {
		Jedis jedis = RedisConnection.getJedis();
		jedis.set(key, JSON.toJSONString(obj));
		jedis.close();
	}
	//获取
	public static String getJsonString(String key) {
		Jedis jedis = RedisConnection.getJedis();
		String value = jedis.get(key);
		jedis.close();
		return value;
	}

这里是使用fastjson的相关函数toJSONString将对象转换为字符串进行存储。获取的时候直接返回json字符串给前端就可以了。使用这种方式可能只能存储简单的json字符串,对于复杂格式的可能会解析错误。

2. 使用对象序列化方式存储

先将对象以字节序列化存储,然后再反序列化得到对象,这里可以封装一个序列化和反序列化的工具类:

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

public class SerializeUtil {
    /*
     * 序列化
     * */
    public static byte[] serizlize(Object object){
        ObjectOutputStream oos = null;
        ByteArrayOutputStream baos = null;
        try {
            baos = new ByteArrayOutputStream();
            oos = new ObjectOutputStream(baos);
            oos.writeObject(object);
            byte[] bytes = baos.toByteArray();
            return bytes;
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            try {
                if(baos != null){
                    baos.close();
                }
                if (oos != null) {
                    oos.close();
                }
            } catch (Exception e2) {
                e2.printStackTrace();
            }
        }
        return null;
    }
    /*
     * 反序列化
     * */
    public static Object deserialize(byte[] bytes){
        ByteArrayInputStream bais = null;
        ObjectInputStream ois = null; 
        try{
            bais = new ByteArrayInputStream(bytes);
            ois = new ObjectInputStream(bais);
            return ois.readObject();
        }catch(Exception e){
            e.printStackTrace();
        }finally {
            try {

            } catch (Exception e2) {
                e2.printStackTrace();
            }
        }
        return null;
    }
}

相应存储和获取代码如下:

    public static void setObject(String key,Object object){
        Jedis jedis = RedisConnection.getJedis();
        jedis.set(key.getBytes(), SerializeUtil.serizlize(object));
        jedis.close();
    }
    public static Object getObject(String key){
        Jedis jedis = RedisConnection.getJedis();
        byte[] bytes = jedis.get(key.getBytes());
        jedis.close();
        return SerializeUtil.deserialize(bytes);
    }

使用这种方式可能遇到嵌套的对象或者json数组等不太好解决。

3. 使用hash存储

第三种方式也是解决我问题的一种方式,由于我的json格式数据比较复杂,形如:

{
  "cd": [{"Condition": {...}, segs:[1,2,3]}, { }, ...]
  "rs": {"way": "休宁路", "road":[{},{},..], "segList": [{object}, {}, ...] }
}

我刚开始是直接使用fastjson将其转换为字符串进行存储,但在返回解析时发现格式不太对了。分析其原因可能是json格式太复杂,然后使用redis中hash方式将数据进行分割存储,即cd对应一个key存储,rs为一个key进行存储,将这两部分都放入一个hash中。相应代码如下:

Jedis jedis = RedisConnection.getJedis();
JSONObject res = new JSONObject();	//最终结果
//如果redis中存在,则直接从redis中取,否则计算并存储至redis
if(jedis.exists(lm)) {
	String rs_value = jedis.hget(lm, "rs");
	String cd_value = jedis.hget(lm, "cd");
	res.put("cd", JSONArray.parseArray(cd_value));
	res.put("rs", JSONObject.parseObject(rs_value));
	System.out.println("redis get success");
} else {
	res = computeRes(lm);
	//更新redis
	jedis.hset(lm, "cd", res.getJSONArray("cd").toJSONString());
	jedis.hset(lm, "rs", JSON.toJSONString(res.getJSONObject("rs")));
	System.out.println("redis set success");
}
jedis.close();
		
//候选结果集转json字符串
String jsonStr = JSON.toJSONString(res, SerializerFeature.DisableCircularReferenceDetect);
//返回给前端
System.out.println("json string: " + jsonStr);
response.setContentType("text/html;charset=utf-8");  //解决前端中文乱码
PrintWriter out = response.getWriter();
out.print(jsonStr);

我是以路名(lm)作为hash的key,首先判断是否存在该key,不存在的话先进行计算res = computeRes(lm);,得到上述的json格式数据,然后使用hset方法将其分别作为两个key进行存储。注意上述cd为一个JSONArray对象,需要使用调用toJSONString()方法转换成字符串,而rs为一个JSONObject对象,使用的是JSON.toJSONString
在获取时首先分别获取其字符串形式,然后分别转换成相应的类型JSONArray.parseArray(cd_value)JSONObject.parseObject(rs_value),最终包裹在一个JSONObject中。

参考:

  1. https://blog.youkuaiyun.com/CrazyLai1996/article/details/76472856
  2. http://www.runoob.com/redis/redis-hashes.html
  3. https://blog.youkuaiyun.com/u011008029/article/details/51315339
<think>好的,我需要解决用户的问题:如何在JSON字符串中转义反斜杠\。首先,我得回忆JSON的转义规则。记得JSON中反斜杠是转义字符,所以如果要表示一个反斜杠本身,应该用两个反斜杠\\对吧?比如,用户可能有一个字符串里有路径"C:\Program Files",在JSON中应该写成"C:\\Program Files"。 接下来,我得考虑不同编程语言处理JSON时的具体方法。比如在PHP中,用户可能需要使用stripslashes函数或者json_decode的选项来去除多余的转义。而Java里可能需要用replace方法替换反斜杠。用户提供的引用里提到了这些情况,比如引用3中的Java代码示例,替换所有反斜杠。不过需要注意,可能要根据具体情况处理,避免过度替换。 然后,我需要检查用户提供的引用内容,确保我的回答符合这些资料。引用1提到JSON保存在文件中后读取时不能直接解析,因为反斜杠的问题。引用2和4讨论了PHP和Java中去掉反斜杠的方法。引用5提到处理动态Json字符串时的问题。这些都需要整合到回答中,说明不同语言的处理方式。 可能用户的问题还涉及到生成和解析JSON时的正确转义。比如,在生成JSON时,正确转义反斜杠,确保解析时不会出错。比如,在JavaScript中,使用JSON.stringify会自动处理转义,但手动构造字符串时需要注意。 还要注意,有些情况下,字符串可能被多次转义,导致出现多个反斜杠,这时候需要逐步去除。例如,引用3中的例子,字符串有多层反斜杠,需要用replace方法替换掉。 最后,应该总结步骤:生成JSON时用两个反斜杠表示一个,解析时根据语言处理多余的转义。同时,结合用户提供的引用,给出具体语言的示例,提醒注意多次转义的情况。</think>在JSON中转义反斜杠字符需遵循以下规则: ### 一、JSON转义规则 1. **基础转义**:JSON中每个反斜杠`\`必须用双反斜杠`\\`表示 $$ C:\Program\ Files \rightarrow "C:\\\\Program\\\\Files" $$ 这是JSON规范强制要求的字符转义机制[^1] 2. **组合转义**:遇到特殊字符组合时需完整转义 ```json { "regex": "^\\d{4}$", // 正则表达式^\d{4}$的JSON表示 "path": "D:\\\\data\\\\logs" } ``` ### 二、编程语言处理示例 #### PHP处理流程: ```php // 添加转义 $json = json_encode(['path' => 'C:\Windows'], JSON_UNESCAPED_SLASHES); // 去除转义 $data = json_decode(stripslashes($_POST['json']), true); // 引用[2] ``` #### Java处理方法: ```java // 生成时自动转义 String json = new Gson().toJson(new FilePath("D:\\logs")); // 手动处理转义 String input = "{\\\"key\\\":\\\"value\\\"}"; String clean = input.replace("\\", ""); // 引用[3] ``` #### JavaScript处理: ```javascript // 自动转义 const obj = { path: 'C:\\Program Files' }; const json = JSON.stringify(obj); // 输出:"{\"path\":\"C:\\\\Program Files\"}" // 解析处理 const parsed = JSON.parse('{"path":"C:\\\\Program Files"}'); ``` ### 三、常见问题诊断 1. **双重转义**:常见于多次序列化场景 ```python # 错误示例:经过两次转义 import json data = {'path': r'C:\Windows'} json_str = json.dumps(json.dumps(data)) # 产生双重转义 ``` 2. **文件读写**:从文件读取JSON时需注意编码 ```java // 正确读取方式(Java示例) String content = Files.readString(Paths.get("data.json")); JSONObject obj = new JSONObject(content.replace("\\", "\\\\")); // 引用[5] ```
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值