zoukankan      html  css  js  c++  java
  • Redis系列 | 阿里云 Redis 版数据库Lua脚本支持与限制

    Lua使用限制

    为了保证脚本里面的所有操作都在相同slot进行,云数据库Redis集群版本会对Lua脚本做如下限制:

    所有key都应该由KEYS数组来传递,redis.call/pcall中调用的redis命令,key的位置必须是KEYS array(不能使用Lua变量替换KEYS),否则直接返回错误信息:

    原生的Lua脚本编写

                $script = <<<luascript
    			local rate_key = KEYS[1]
    			local limit = tonumber(ARGV[1]) -- 最大限制
    			local expire_time = ARGV[2] -- 过期时间
    			local result = redis.call('SETNX',rate_key,1); -- SETNX 只有不存在的时候才设置,返回1。否则返回0
    			if result == 1 then
    				redis.call('expire',rate_key,expire_time)
    				return 1
    			else
    				if tonumber(redis.call("GET", rate_key)) >= limit then
    					return 0
    				else
    					redis.call("INCR", rate_key)
    					return 1
    				end
    			end
    luascript;
                $scriptSha = $redis->script('load', $script);
    

     以上脚本在原生的Redis是可以运行的,但是在阿里云Redis则不行,直接返回false

    解释一下官方的限制:

    key的位置必须是KEYS array(不能使用Lua变量替换KEYS),而上面的脚本则是把KEYS[1] 赋值给了Lua变量 rate_key ,从而导致不能使用

    最终修改兼容版本

            local limit = tonumber(ARGV[1]) -- 最大限制
            local expire_time = ARGV[2] -- 过期时间
            local result = redis.call('SETNX',KEYS[1],1); -- SETNX 只有不存在的时候才设置,返回1。否则返回0
            if result == 1 then
                redis.call('expire',KEYS[1],expire_time)
                return 1
            else
                if tonumber(redis.call("GET", KEYS[1])) >= limit then
                    return 0
                else
                    redis.call("INCR", KEYS[1])
                    return 1
                end
            end
    luascript;
            $scriptSha = $redis->script('load', $script);
    

      

     

    
    
  • 相关阅读:
    UVA
    剑指offer 面试题11
    SharePoint 2013+ Sqlserver 2014 Kerberos 配置传奇, 最终的解决方案 验证。
    ASC(1)G(上升时间最长的序列)
    J2EE它是一个框架?平台?规范?
    C++和python使用struct传输二进制数据结构来实现
    前端开发面试题集(二)
    C语言中<CR>是什么意思
    Delphi ParamStr 使用方法
    打包工具 使用帮助 inno setup
  • 原文地址:https://www.cnblogs.com/tinywan/p/13064510.html
Copyright © 2011-2022 走看看