当前位置: 首页> 新闻> 资讯 > seo技术外包 乐云践新专家_创造一个平台要多少钱_竞价托管 微竞价_拉新推广

seo技术外包 乐云践新专家_创造一个平台要多少钱_竞价托管 微竞价_拉新推广

时间:2025/9/12 12:38:26来源:https://blog.csdn.net/qq_62877881/article/details/147360821 浏览次数:0次
seo技术外包 乐云践新专家_创造一个平台要多少钱_竞价托管 微竞价_拉新推广

一、Lua脚本操作Redis的优势

特性说明
原子性Lua脚本在Redis中单线程执行,所有操作要么全部成功,要么全部失败。
减少网络开销将多个Redis命令合并为一个脚本执行,减少客户端与服务端的通信次数
复杂逻辑封装可实现条件判断、循环、计算等复杂逻辑(Redis原生命令无法直接实现)。
集群兼容性通过显式声明KEYS,保证在Redis集群模式下正确路由到目标节点。

二、Lua脚本基本规范

1. 参数传递
  • KEYS数组:所有操作的Redis键(Key)必须显式声明,通过KEYS[1], KEYS[2]访问。
  • ARGV数组:其他参数(如值、标志位)通过ARGV[1], ARGV[2]访问。
  • 示例
    -- KEYS[1]=user:123:likes, ARGV[1]=pic_456
    redis.call('SADD', KEYS[1], ARGV[1])
    
2. 返回值
  • Lua脚本的最后一个值会作为执行结果返回给客户端。
  • 可返回nil、数值、字符串、表(自动转为Redis多行回复)。
3. 脚本编写原则
  • 禁止使用全局变量:所有变量需用local声明。
  • 避免长耗时操作:Lua脚本会阻塞Redis其他请求,需确保高效性。

三、常用Lua脚本操作

1. 数据操作
操作示例代码
Stringredis.call('SET', KEYS[1], ARGV[1])
Hashredis.call('HSET', KEYS[1], 'field', ARGV[1])
Setredis.call('SADD', KEYS[1], ARGV[1])
ZSetredis.call('ZADD', KEYS[1], ARGV[1], ARGV[2])
Listredis.call('LPUSH', KEYS[1], ARGV[1])
2. 条件判断
if redis.call('EXISTS', KEYS[1]) == 1 thenreturn redis.call('GET', KEYS[1])
elsereturn nil
end
3. 循环操作
for i=1, #ARGV doredis.call('SADD', KEYS[1], ARGV[i])
end
4. 错误处理
local ok, err = pcall(redis.call, 'INCRBY', KEYS[1], ARGV[1])
if not ok thenreturn {err = err}
end

四、注意事项

场景解决方案
集群模式确保所有KEYS在同一个哈希槽(可通过Hash Tag实现,如{user}:123:likes)。
脚本性能避免复杂循环或大范围数据遍历,优先用Redis原生命令。
脚本缓存使用SCRIPT LOAD预加载脚本,通过EVALSHA执行(减少网络传输)。
调试脚本通过redis.log(redis.LOG_DEBUG, 'message')输出日志(需配置Redis日志级别)。

五、典型应用场景

场景Lua脚本作用
分布式锁原子化实现锁的获取、续期、释放(避免锁误删)。
计数器原子化增减计数(如点赞数、库存扣减)。
排行榜计算分数并更新ZSet,返回排名结果。
批量操作合并多个命令(如先检查条件再删除数据)。

六、调试与测试

  1. 直接执行脚本(通过redis-cli):
    redis-cli --eval script.lua key1 key2 , arg1 arg2
    
  2. 捕获错误
    local ok, result = pcall(redis.call, 'COMMAND', params)
    if not ok thenreturn {error = result}
    end
    

点赞实现

编写lua脚本

public class RedisLuaScript {/*** 点赞 Lua 脚本* KEYS[1]       -- 临时计数键* KEYS[2]       -- 用户点赞状态键* ARGV[1]       -- 用户 ID* ARGV[2]       -- 壁纸 ID* 返回:* -1: 已点赞* 1: 操作成功*/public static final RedisScript<Long> LIKE_SCRIPT = new DefaultRedisScript<>("local tempLikeKey = KEYS[1]\n" +"local userLikeKey = KEYS[2]\n" +"local userId = ARGV[1]\n" +"local picId = ARGV[2]\n" +"\n" +"-- 1. 检查是否已点赞(避免重复操作)\n" +"if redis.call('HEXISTS', userLikeKey, picId) == 1 then\n" +"    return -1\n" +"end\n" +"\n" +"-- 2. 获取旧值(不存在则默认为 0)\n" +"local hashKey = userId .. ':' .. picId\n" +"local oldNumber = tonumber(redis.call('HGET', tempLikeKey, hashKey) or 0)\n" +"\n" +"-- 3. 计算新值\n" +"local newNumber = oldNumber + 1\n" +"\n" +"-- 4. 原子性更新:写入临时计数 + 标记用户已点赞\n" +"redis.call('HSET', tempLikeKey, hashKey, newNumber)\n" +"redis.call('HSET', userLikeKey, picId, 1)\n" +"\n" +"return 1", Long.class);/*** 取消点赞 Lua 脚本* 参数同上* 返回:* -1: 未点赞* 1: 操作成功*/public static final RedisScript<Long> UNLIKE_SCRIPT = new DefaultRedisScript<>("local tempLikeKey = KEYS[1]\n" +  // 显式换行"local userLikeKey = KEYS[2]\n" +"local userId = ARGV[1]\n" +"local picId = ARGV[2]\n" +"\n" +  // 空行分隔逻辑块"-- 1. 检查用户是否已点赞(若未点赞,直接返回失败)\n" +"if redis.call('HEXISTS', userLikeKey, picId) ~= 1 then\n" +"    return -1\n" +"end\n" +"\n" +"-- 2. 获取当前临时计数(若不存在则默认为 0)\n" +"local hashKey = userId .. ':' .. picId\n" +"local oldNumber = tonumber(redis.call('HGET', tempLikeKey, hashKey) or 0)\n" +"\n" +"-- 3. 计算新值并更新\n" +"local newNumber = oldNumber - 1\n" +"\n" +"-- 4. 原子性操作:更新临时计数 + 删除用户点赞标记\n" +"redis.call('HSET', tempLikeKey, hashKey, newNumber)\n" +"redis.call('HDEL', userLikeKey, picId)\n" +"\n" +  // 确保return前有换行"return 1",  // 最后一行无需\n(Redis会自动补全)Long.class);
}

执行lua脚本

redisTemplate.execute(RedisLuaScriptConstant.LIKE_SCRIPT,Arrays.asList(tempLikeKey,userLikeKey),loginUser.getId(),picId
);
redisTemplate.execute(RedisLuaScriptConstant.LIKE_SCRIPT,Arrays.asList(tempLikeKey,userLikeKey),loginUser.getId(),picId
);
关键字:seo技术外包 乐云践新专家_创造一个平台要多少钱_竞价托管 微竞价_拉新推广

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

责任编辑: