RedisTemplate调用lua踩坑了,差点导致项目delay

RedisTemplate的Serializer将不同的数据类型进行序列化时,内置了一套逻辑,譬如Long类型的追回“L”,Byte的追加“B”。这就可能与redis期望的数据不一致,而引发错误

背景

要实现 次数/统计周期 的效果,可以通过incr和expire命令来实现:

1、根据访问者ip生成Redis key

2、执行incr命令,将key对应的数字+1

3、执行incr命令执行结果为1时【当前统计周期内第一次访问】,使用expire命令将key的过期时间设置为统计周期

4、根据incr的结果判断是否超出访问频率

看看上面的线程安全问题,会有啥影响

不过,上面的实现方案,在并发时,会有线程安全问题。【没get的小伙伴,可以留言讨论】

然后,查看了redis的文档时看到了建议的解法:

Lua脚本实现自增并过期

https://redis.io/commands/incr/

local current
current = redis.call("incr",KEYS[1])
if current == 1 then
    redis.call("expire",KEYS[1],1)
end

问题描述

使用RedisTemplate执行上面的lua脚本是可以的。

但是有个问题:key的过期时间hard code在lua脚本中了。

统计需要根据情况动态调整的。

把 统计周期 作为参数传入时,却报错了:

ERR value is not an integer or out of range

show the code :

@SpringBootTest(classes = MainApplication.class)
@RunWith(Spri
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: redistemplate可以执行lua脚本,可以通过以下步骤实现: 1. 创建一个RedisTemplate对象。 2. 通过RedisTemplate对象获取一个RedisConnection对象。 3. 通过RedisConnection对象执行lua脚本,可以使用RedisConnection的eval()方法。 4. eval()方法的参数包括lua脚本和脚本中需要的参数。 5. 执行完毕后,可以通过eval()方法的返回值获取脚本执行结果。 例如: ``` RedisTemplate<String, String> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(redisConnectionFactory); RedisConnection connection = redisTemplate.getConnectionFactory().getConnection(); String script = "return redis.call('get', KEYS[1])"; String result = connection.eval(script.getBytes(), ReturnType.VALUE, 1, "key".getBytes()); ``` 以上代码执行了一个简单的lua脚本,获取了Redis中key为"key"的值。 ### 回答2: redis是一个内存缓存数据库,可用于提高Web应用程序的性能。而lua是一种脚本语言,可以被嵌入到各种编程语言中,包括redis。redistemplate是redis的一个Java客户端,可以用于与redis进行交互。redistemplate提供了一个execute方法,用于执行lua脚本。 执行lua脚本可以有效地提高redis的性能。在执行许多redis命令的情况下,每个命令都需要进行网络延迟,这会对性能产生负面影响。但是,通过执行lua脚本,可以将多个redis命令打包在一起,从而减少网络延迟。此外,lua脚本还可以在服务端执行,从而减少客户端和服务端之间的数据传输量。 使用redistemplate执行lua脚本可以分为以下几步: 1.创建lua脚本 在执行lua脚本之前,需要创建脚本。lua脚本是一段文本,其语法类似于其他脚本语言。在编写脚本时,需要使用redis提供的一些特殊命令,例如get/set等。这些命令与普通的redis命令略有不同,需要使用redis.call或redis.pcall等特殊命令来调用。此外,还可以定义一些函数,从而使脚本更具可读性。 2.使用redistemplate执行脚本 redistemplate提供了execute方法,用于执行lua脚本。execute方法有两个参数:RedisScript对象和RedisSerializer对象。RedisScript对象表示要执行的lua脚本,可以使用LuaScriptBuilder类来构建。RedisSerializer对象表示要使用的序列化程序,用于将参数转换为字节数组以及将结果转换回对象。默认情况下,redistemplate使用JdkSerializationRedisSerializer来序列化对象,但也可以使用其他序列化程序,例如Jackson2JsonRedisSerializer。 执行lua脚本的结果是一个Object类型的值,可以通过类型转换将其转换为所需的类型,例如String、List等。此外,还可以使用redis提供的一些命令获取更复杂的结果,例如evalsha命令可以获取lua脚本的SHA1哈希值,从而可以在之后的操作中更高效地使用这个脚本。 总的来说,通过使用redistemplate执行lua脚本,可以在redis中实现更复杂的操作,从而提高应用程序的性能和可靠性。虽然lua脚本相对复杂一些,但学习好后会发现有很多用处。 ### 回答3: redistemplate是Redis中的一个模板,在执行Lua脚本时可以使用它来提供更多的功能。 在Redis中使用Lua脚本可以实现一些比较复杂的操作,而redistemplate可以为这些操作提供更好的支持。例如,当需要操作Redis的键和值时,通常需要先对它们进行序列化和反序列化,这个过程比较繁琐,但redistemplate提供了这些操作的封装,使得操作起来更为简单。 使用redistemplate执行Lua脚本时,需要将脚本和参数都传递给redistemplate,然后由redistemplate将它们交给Redis进行执行。在执行完成后,redistemplate会返回脚本的执行结果,同时也会释放与此脚本相关的所有资源。 需要注意的是,在使用redistemplate执行Lua脚本时,应该确保脚本的安全性。因为Lua脚本具有比较强的执行能力,如果脚本存在安全漏洞,可能会导致Redis的数据丢失或其他安全问题。因此,在编写和执行Lua脚本时应该遵守最佳安全实践,避免脚本的恶意使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

也和光同尘

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值