EVAL、EVALSHA命令
Redis
从2.6.0版本开始提供了eval
命令,通过内置的Lua
解释器,可以让用户执行一段Lua
脚本并返回数据。因为Redis
单线程模型的特点,可以保证多个命令的原子性
(因为最近的项目才想到用Lua),详细的使用方法请移步官方文档。
脚本性能
-
Redis
保证了脚本执行的原子性,所以在当前脚本没执行完之前,别的命令和脚本都是等待状态,所以一定要控制好脚本中的内容,防止出现需要消耗大量时间的内容(逻辑相对简单)。
带宽优化
- 为了避免每次执行都重复的将
Lua
脚本内容发送,Redis
提供了evalsha
命令,只需要将Lua脚本内容的SHA1校验和发送即可(evalsha 6b1bf486c81ceb7edf3c093f4c48582e38c0e791 0
)。 - Lua脚本中的
变量
(动态数据)请使用KEYS
和ARGV
获取,如果把变量
放在脚本中,必然会导致每次的脚本内容都不同(SHA1),Redis缓存大量无用或者一次性的脚本内容。
Redis Cluster 或 阿里云Redis集群版使用注意事项
Redis从3.0开始支持了Cluster功能,之前使用eval
的时候可能没什么问题,但当切换成Cluster模式的时候,可能会出现一些问题:
- ERR Error running script (call to f_4a610f5543b3c3450220da7bd47825d3b6bffae8): @user_script:1: @user_script: 1: Lua script attempted to access a non local key in a cluster node
- ERR eval/evalsha command keys must be in same slot(阿里云Redis集群版)
上面的错误是因为Redis要求单个Lua脚本操作的key必须在同一个节点上,但是Cluster会将数据自动分布到不同的节点(虚拟的16384个slot,具体看官方文档),阿里云集群版的官网其实也有对应说明:在Redis集群版实例中,事务、脚本等命令要求所有的key必须在同一个slot中,如果不在同一个slot中将返回以下错误信息(:command keys must in same slot)
如何解决?
CLUSTER KEYSLOT key的文档中提供了解决方法,你需要将把key中的一部分使用{}
包起来,redis将通过{}
中间的内容作为计算slot的key,类似key1{mykey}
、key2{mykey}
这样的都会存放到同一个slot中(缺点是不能平滑的过度老业务,需要修改原来使用的key,如果之前的key是统一管理的,也没那么麻烦)
思考
如果某个业务都通过key{mykey}
去储存获取内容,所有的操作都会hash到同一个slot,这个slot所在的节点压力就会变大(不均衡),如果解决?欢迎留言讨论~