使用Lua脚本的好处
1、减少网络开销:可以将多个请求通过脚本的形式一次发送,减少网络时延和请求次数。
2、原子性的操作:Redis会将整个脚本作为一个整体执行,中间不会被其他命令插入。因此在编写脚本的过程中无需担心会出现竞态条件,无需使用事务。
3、代码复用:客户端发送的脚步会永久存在redis中,这样,其他客户端可以复用这一脚本来完成相同的逻辑。
4、速度快:见 与其它语言的性能比较, 还有一个 JIT编译器可以显著地提高多数任务的性能; 对于那些仍然对性能不满意的人, 可以把关键部分使用C实现, 然后与其集成, 这样还可以享受其它方面的好处。
5、可以移植:只要是有ANSI C 编译器的平台都可以编译,你可以看到它可以在几乎所有的平台上运行:从 Windows 到Linux,同样Mac平台也没问题, 再到移动平台、游戏主机,甚至浏览器也可以完美使用 (翻译成JavaScript).
6、源码小巧:20000行C代码,可以编译进182K的可执行文件,加载快,运行快。
redis为什么需要lua呢?简单来说为了性能以及事务的原子性
1.redis lua 减少库存操作类
Component
@Slf4j
public class RedisDecr {
@Autowired
private StringRedisTemplate redisTemplate;
/**
* 递减
* @param key
* @param value
* @return -1失败,其他值为递减后的值
*/
public Integer decr(String key, String value) {
return tryDecrPackage(key,value);
}
private Integer tryDecrPackage(String key, String value){
DefaultRedisScript<String> defaultRedisScript = new DefaultRedisScript<>();
defaultRedisScript.setResultType(String.class);
defaultRedisScript.setScriptText("if redis.call('decrby',KEYS[1],ARGV[1]) < 0 then redis.call('incrby',KEYS[1],ARGV[1]) return '-1' else return redis.call('GET', KEYS[1]) end");
List<String> keys = new ArrayList<>();
keys.add(key);
String result = redisTemplate.execute(defaultRedisScript, keys, value);
log.info("key:{} value:{} 脚本返回结果为:{}",key,value,result);
return Integer.parseInt(result);
}
}
2.调用减库存
Integer decr = redisDecr.decr(PuhuiyunRedisKeyConstants.EXCHANGE_HANDLE_REMAIN_STOCK_KEY,String.valueOf(amount));
3.增加库存
Long increment = this.stringRedisTemplate.opsForValue().increment(PuhuiyunRedisKeyConstants.EXCHANGE_HANDLE_REMAIN_STOCK_KEY, amount);