hi,你好!欢迎访问本站!登录
本站由简数采集腾讯云宝塔系统阿里云强势驱动
当前位置:首页 - 教程 - 杂谈 - 正文 君子好学,自强不息!

SpringBoot + Redis 实行lua剧本

2019-11-18杂谈搜奇网41°c
A+ A-

1、背景

有时刻,我们须要一次性操纵多个 Redis 敕令,然则 如许的多个操纵不具备原子性,而且 Redis 的事件也不够壮大,不支撑事件的回滚,还没法完成敕令之间的逻辑关系盘算。所以,平常在开辟中,我们会应用 lua 脚原本完成 Redis 的事件。

2、lua 剧本

Redis 中运用 lua 剧本,我们须要注重的是,从 Redis 2.6.0后才支撑 lua 剧本的实行。
运用 lua 剧本的优点:

  • 原子操纵:lua剧本是作为一个团体实行的,所以中心不会被其他敕令插进去。
  • 削减收集开支:能够将多个要求经由过程剧本的情势一次发送,削减收集时延。
  • 复用性:lua剧本能够常驻在redis内存中,所以在运用的时刻,能够直接拿来复用,也削减了代码量。

3、Redis 中实行 lua 剧本

1、敕令花样:

EVAL script numkeys key [key ...] arg [arg ...]

申明:

  • script是第一个参数,为Lua 5.1剧本(字符串)。
  • 第二个参数numkeys指定后续参数有几个key。
  • key [key ...],被操纵的key,能够多个,在lua剧本中经由过程KEYS[1], KEYS[2]猎取
  • arg [arg ...],参数,能够多个,在lua剧本中经由过程ARGV[1], ARGV[2]猎取。

2、假如直接运用 redis-cli敕令:

redis-cli --eval lua_file key1 key2 , arg1 arg2 arg3

申明:

  • eval 敕令后不再是 lua 剧本的字符串情势,而是一个 lua 剧本文件。后缀为.lua
  • 不再须要numkeys参数,而是用 , 离隔多个key和多个arg

4、运用 RedisTemplate 实行 lua 剧本

例子:删除 Redis 分布式锁
引入依靠:此依靠为我们整合了 Redis ,而且供应了异常好用的 RedisTemplate。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

体式格局一:lua 剧本文件
1、新建 lua 剧本文件:

if redis.call("get",KEYS[1]) == ARGV[1] then
    return redis.call("del",KEYS[1])
else
    return 0
end

申明:先猎取指定key的值,然后和传入的arg比较是不是相称,相称值删除key,不然直接返回0。

2、代码测试:

/**
 * @author Howinfun
 * @desc lua 测试
 * @date 2019/11/5
 */
@RunWith(SpringJUnit4ClassRunner.class)
@ActiveProfiles("test")
@SpringBootTest(classes = ThirdPartyServerApplication.class)
public class RedisTest {

    @Autowired
    private StringRedisTemplate redisTemplate;
    @Test
    public void contextLoads() {
        String lockKey = "123";
        String UUID = cn.hutool.core.lang.UUID.fastUUID().toString();
        boolean success = redisTemplate.opsForValue().setIfAbsent(lockKey,UUID,3, TimeUnit.MINUTES);
        if (!success){
            System.out.println("锁已存在");
        }
        // 实行 lua 剧本
        DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();
        // 指定 lua 剧本
        redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource("redis/DelKey.lua")));
        // 指定返回范例
        redisScript.setResultType(Long.class);
        // 参数一:redisScript,参数二:key列表,参数三:arg(可多个)
        Long result = redisTemplate.execute(redisScript, Collections.singletonList(lockKey),UUID);
        System.out.println(result);
    }
}

体式格局二:直接编写 lua 剧本(字符串)
1、代码测试:

/**
 * @author Howinfun
 * @desc lua 剧本测试
 * @date 2019/11/5
 */
@RunWith(SpringJUnit4ClassRunner.class)
@ActiveProfiles("test")
@SpringBootTest(classes = ThirdPartyServerApplication.class)
public class RedisTest {

    /** 开释锁lua剧本 */
    private static final String RELEASE_LOCK_LUA_SCRIPT = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";

    @Autowired
    private StringRedisTemplate redisTemplate;
    @Test
    public void contextLoads() {
        String lockKey = "123";
        String UUID = cn.hutool.core.lang.UUID.fastUUID().toString();
        boolean success = redisTemplate.opsForValue().setIfAbsent(lockKey,UUID,3, TimeUnit.MINUTES);
        if (!success){
            System.out.println("锁已存在");
        }
        // 指定 lua 剧本,而且指定返回值范例
        DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>(RELEASE_LOCK_LUA_SCRIPT,Long.class);
        // 参数一:redisScript,参数二:key列表,参数三:arg(可多个)
        Long result = redisTemplate.execute(redisScript, Collections.singletonList(lockKey),UUID);
        System.out.println(result);
    }
}

注重:能够会有同砚发明,为何返回值不必 Integer 吸收而是用 Long。这里是因为 spring-boot-starter-data-redis 供应的返回范例内里不支撑 Integer。
人人能够看看源码:

/**
 * Represents a data type returned from Redis, currently used to denote the expected return type of Redis scripting
 * commands
 *
 * @author Jennifer Hickey
 * @author Christoph Strobl
 */
public enum ReturnType {

    /**
     * Returned as Boolean
     */
    BOOLEAN,

    /**
     * Returned as {@link Long}
     */
    INTEGER,

    /**
     * Returned as {@link List<Object>}
     */
    MULTI,

    /**
     * Returned as {@literal byte[]}
     */
    STATUS,

    /**
     * Returned as {@literal byte[]}
     */
    VALUE;

    /**
     * @param javaType can be {@literal null} which translates to {@link ReturnType#STATUS}.
     * @return never {@literal null}.
     */
    public static ReturnType fromJavaType(@Nullable Class<?> javaType) {

        if (javaType == null) {
            return ReturnType.STATUS;
        }
        if (javaType.isAssignableFrom(List.class)) {
            return ReturnType.MULTI;
        }
        if (javaType.isAssignableFrom(Boolean.class)) {
            return ReturnType.BOOLEAN;
        }
        if (javaType.isAssignableFrom(Long.class)) {
            return ReturnType.INTEGER;
        }
        return ReturnType.VALUE;
    }
}

所以当我们运用 Integer 作为返回值的时刻,是报以下毛病:

org.springframework.data.redis.RedisSystemException: Redis exception; nested exception is io.lettuce.core.RedisException: java.lang.IllegalStateException

    at org.springframework.data.redis.connection.lettuce.LettuceExceptionConverter.convert(LettuceExceptionConverter.java:74)
    at org.springframework.data.redis.connection.lettuce.LettuceExceptionConverter.convert(LettuceExceptionConverter.java:41)
  选择打赏方式
微信赞助

打赏

QQ钱包

打赏

支付宝赞助

打赏

  移步手机端
SpringBoot + Redis 实行lua剧本

1、打开你手机的二维码扫描APP
2、扫描左则的二维码
3、点击扫描获得的网址
4、可以在手机端阅读此文章
未定义标签

本文来源:搜奇网

本文地址:https://www.sou7.cn/282375.html

关注我们:微信搜索“搜奇网”添加我为好友

版权声明: 本文仅代表作者个人观点,与本站无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。请记住本站网址https://www.sou7.cn/搜奇网。

发表评论

选填

必填

必填

选填

请拖动滑块解锁
>>