参考redis写分布式竞争一个锁的时候,碰到了lua脚本的问题,引出学习一下
参考文档:
redis.io
https://wudashan.cn/2017/10/23/Redis-Distributed-Lock-Implement/
由于涉及到lua脚本,所以又简单学习了一下 redis的lua脚本
参考文档:http://redisdoc.com/script/eval.html
Lua: 给 Redis 用户的入门指导
一、为什么要用脚本
减少网络开销。可以将多个请求通过脚本的形式一次发送,减少网络时延
原子操作。redis会将整个脚本作为一个整体执行,中间不会被其他命令插入。因此在编写脚本的过程中无需担心会出现竞态条件,无需使用事务。
复用。客户端发送的脚步会永久存在redis中,这样,其他客户端可以复用这一脚本而不需要使用代码完成相同的逻辑。
二、如何使用脚本
这里的重点是关于 keys和argv的理解,因为所有的网站举例都是
- script : 参数是一段 Lua 5.1 脚本程序。脚本不必(也不应该)定义为一个 Lua 函数。
- numkeys : 用于指定键名参数的个数。
- key [key ...] : 从 EVAL 的第三个参数开始算起,表示在脚本中所用到的那些 Redis 键(key),这些键名参数可以在 Lua 中通过全局变量 KEYS 数组,用 1 为基址的形式访问( KEYS[1] , KEYS[2] ,以此类推)。
- arg [arg ...] : 附加参数,在 Lua 中通过全局变量 ARGV 数组访问,访问的形式和 KEYS 变量类似( ARGV[1] 、 ARGV[2] ,诸如此类)。
格式:
redis 127.0.0.1:6379> EVAL script numkeys key [key ...] arg [arg ...]
举例:
redis 127.0.0.1:6379> eval "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}" 2 key1 key2 first second
1) "key1"
2) "key2"
3) "first"
4) "second"
这个举例其实完全看不出KEYS和ARGV参数的区别,明明一个是实际的键名,一个是附属的参数,换个例子就很好理解
redis>EVAL "return redis.call('SET', KEYS[1], ARGV[1])" 1 foo bar
OK
redis>GET foo
"bar"
- redis lua 脚本所有的键都应该用keys传递
网友评论