【Redis】问题记录:redisTemplate调用Lua脚本获取返回值时报错(Could not read JSON:Unrecognized token .....)解决办法

阅读: 评论:0

【Redis】问题记录:redisTemplate调用Lua脚本获取返回值时报错(Could not read JSON:Unrecognized token .....)解决办法

【Redis】问题记录:redisTemplate调用Lua脚本获取返回值时报错(Could not read JSON:Unrecognized token .....)解决办法

文章目录

  • 文章目录
  • 正文
  • 一、问题概述:
  • 二、原因分析:
  • 三、解决方案:
  •         方案一(掩耳盗铃法 -- 推荐)
  •         方案二(修改序列化器 -- 不推荐)
  •         方案三 (使用StringRedisTemplate -- 推荐) 

正文

        redisTemplate配置可以看我的这篇文章:
【Redis】RedisTemplate序列化配置、解决LocalDateTime数据类型序列化问题


一、问题概述:

        在使用了GenericJackson2JsonRedisSerializer配置RedisTemplate的序列化之后,当调用Lua脚本并获取字符串类型的返回值时却报错:

        org.dis.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.dis.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 = ute(script, ptyList());System.out.println(execute);
}
return "test" --运行失败
return 1111 -- 运行成功
return redis.call('get','mykey') -- 运行成功

二、原因分析:

        根据报错信息SerializationException: Could not read JSON:Unrecognized token 'test'以及后面的at org.dis.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 = ute(script, ptyList());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 = ute(script, ptyList());System.out.println(execute);
}

 完美运行! 当然@Autowired也可以改为:

@Autowired
private RedisTemplate stringRedisTemplate;

本文发布于:2024-01-28 07:03:07,感谢您对本站的认可!

本文链接:https://www.4u4v.net/it/17063966105655.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

标签:解决办法   脚本   返回值   时报   Lua
留言与评论(共有 0 条评论)
   
验证码:

Copyright ©2019-2022 Comsenz Inc.Powered by ©

网站地图1 网站地图2 网站地图3 网站地图4 网站地图5 网站地图6 网站地图7 网站地图8 网站地图9 网站地图10 网站地图11 网站地图12 网站地图13 网站地图14 网站地图15 网站地图16 网站地图17 网站地图18 网站地图19 网站地图20 网站地图21 网站地图22/a> 网站地图23