文章目录
正文
redisTemplate配置可以看我的这篇文章:
【Redis】RedisTemplate序列化配置、解决LocalDateTime数据类型序列化问题https://blog.csdn.net/zxy2361380031/article/details/134121106
一、问题概述:
在使用了GenericJackson2JsonRedisSerializer配置RedisTemplate的序列化之后,当调用Lua脚本并获取字符串类型的返回值时却报错:
org.springframework.data.redis.serializer.SerializationException: Could not read JSON:Unrecognized token 'test': was expecting (JSON String, Number, Array, Object or token 'null', 'true' or 'false') at [Source: (byte[])"test"; line: 1, column: 5] at org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer.deserialize(GenericJackson2JsonRedisSerializer.java:253)
而当Lua脚本的返回值为其他类型,或为使用redis.call()命令查询出的字符串时就能正常运行
Java代码和Lua脚本分别为:
@Autowired
private RedisTemplate redisTemplate;
@Test
void luaTest() {
DefaultRedisScript<Object> script = new DefaultRedisScript<>();
script.setLocation(new ClassPathResource("lua/test.lua"));
script.setResultType(Object.class);
Object execute = redisTemplate.execute(script, Collections.emptyList());
System.out.println(execute);
}
return "test" --运行失败
return 1111 -- 运行成功
return redis.call('get','mykey') -- 运行成功
二、原因分析:
根据报错信息SerializationException: Could not read JSON:Unrecognized token 'test'以及后面的at org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer.deserialize不难得出是由于RedisTemplate的序列化错误而导致的。
由于上面Lua脚本返回的字符串"test"不满足JSON格式,因此GenericJackson2JsonRedisSerializer序列化器在反序列化时会抛出异常。而1111不是字符串,而接收的类型为Object.class,因此会被序列化器默认序列化为Long型数据,从而正常运行。
三、解决方案:
知道了原因,那么解决办法也很明显,下面提供几种解决方法:
方案一(掩耳盗铃法 -- 推荐)
编写Lua脚本时不让它返回字符串不就好了吗,一般用到Redis Lua脚本的业务逻辑常常都是利用redis的原子操作对多个不同Key储存的Value进行修改,当执行条件不满足时,返回一个Long型数据,我们再自己验证并处理就好了,干嘛返回个字符串嘛qwq。总之,尽量不要在Lua脚本中返回字符串。当然由于特殊的业务需要,不得不返回字符串时,可采取下面的方式。
方案二(修改序列化器 -- 不推荐)
当我们需要调用lua脚本时使用redisTemplate.setValueSerializer
(new StringRedisSerializer());修该默认的值序列化器,执行完Lua脚本时再改回来。(因为这个设置会全局修改序列化器)代码如下:
@Autowired
private RedisTemplate redisTemplate;
@Test
void luaTest() {
DefaultRedisScript<Object> script = new DefaultRedisScript<>();
script.setLocation(new ClassPathResource("lua/test.lua"));
script.setResultType(Object.class);
redisTemplate.setValueSerializer(new StringRedisSerializer()); // 修改为string序列化器
Object execute = redisTemplate.execute(script, Collections.emptyList());
redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer()); // 改回Json序列化器
System.out.println(execute);
}
方案三 (使用StringRedisTemplate -- 推荐)
我们的RedisTemplate配置了GenericJackson2JsonRedisSerializer Json序列化器报错了,为何不换一个呢,但像方案二那样非常的麻烦。那有没有默认序列化器就为string类型的类呢,答案是有的。SpringBoot的spring-boot-starter-data-redis为我们提供一个类StringRedisTemplate,正好能够满足我们的需求,使用方法如下:
@Autowired
private StringRedisTemplate stringRedisTemplate;
@Test
void luaTest() {
DefaultRedisScript<Object> script = new DefaultRedisScript<>();
script.setLocation(new ClassPathResource("lua/test.lua"));
script.setResultType(Object.class);
Object execute = stringRedisTemplate.execute(script, Collections.emptyList());
System.out.println(execute);
}
完美运行! 当然@Autowired也可以改为:
@Autowired
private RedisTemplate stringRedisTemplate;