记得上半年读书,看到Lua脚本的时候就跳过了。因为项目用不到,也因为自己技术视野不够宽阔。后来看到一篇介绍秒杀技术的文章,利用Lua脚本保证原子性,这才引起自己的重视。
Redis单个命令都是原子操作,如果想让多个命令聚合成一个原子操作怎么办呢?
- 事务
- Lua脚本
Redis内部内嵌了Lua环境(包含解释器),在启动时会对Lua环境执行一系列修改,从而确保其满足redis在功能性和安全性的需求。
执行Lua脚本命令有:EVAL 和 EVALSHA。
EVAL script numkeys key [key ...] arg [arg ...]
> eval "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}" 2 key1 key2 first second
1) "key1"
2) "key2"
3) "first"
4) "second"
在 Lua 脚本中,可以使用两个不同函数来执行 Redis 命令,它们分别是:
- redis.call()
- redis.pcall()
这两个函数的唯一区别在于它们使用不同的方式处理执行命令所产生的错误。
redis会创建一个lua_scripts字典,key是Lua脚本的SHA1校验和(checksum),value是SHA1校验和对应的Lua脚本。
EVALSHA sha1 numkeys key [key ...] arg [arg ...]
> set foo bar
OK
> eval "return redis.call('get','foo')" 0
"bar"
> evalsha 6b1bf486c81ceb7edf3c093f4c48582e38c0e791 0
"bar"
使用EVALSHA可以减小传播的数据,减小带宽的损耗。
主服务器复制eval命令和复制其他redis命令一样,都是把相同的命令直接传播给从服务器。
Redis 提供了以下几个 SCRIPT 命令,用于对脚本子系统(scripting subsystem)进行控制:
- SCRIPT FLUSH :清除所有脚本缓存
- SCRIPT EXISTS :根据给定的脚本校验和,检查指定的脚本是否存在于脚本缓存
- SCRIPT LOAD :将一个脚本装入脚本缓存,但并不立即运行它
- SCRIPT KILL :杀死当前正在运行的脚本
2019-07-24
子曰:温故而知新,可以为师矣。这个Lua以前怎么费力记忆都会逐渐忘记,但是多看几遍就会有更深入的认识,自然而然就记住了。
旧书不厌百回读,熟读深思子自知。所以,学习是分回合的。
参考资料:《Redis设计与实现》