Redis的使用工具方法类以及Redis锁的使用

 Redis在实际工作的几种使用方法,其中最吸引笔者的是Redis在分布式锁方面的应用。在第二段代码中我会贴出笔者在练习中苏哦用的代码


/**
 * 功能描述: Redis的几种常用工具方法,其中Redis分布式锁注意setIfAbsent方法的使用
 * @param:  
 * @return: 
 * @auther: Rhine
 * @date: 2018/11/12 23:42
 */
public class RedisUtil {

    @Autowired
    private RedisTemplate redisTemplate;



    /**
     * 批量删除对应的value
     * @param keys
     */
    public void remove(final String... keys){
        for(String key:keys){
            remove(key);
        }
    }


    /**
     * 批量删除key
     * @param pattern
     */
    public void removePattern(final String pattern){
        Set<Serializable> keys=redisTemplate.keys(pattern);
        if (keys.size()>0){
            redisTemplate.delete(keys);
        }
    }


    /**
     * 删除对应的value
     * @param key
     */
    public void remove(final String key){
        if (exists(key)){
            redisTemplate.delete(key);
        }
    }


    /**
     * 判断缓存中是否有对应的value
     * @param key
     * @return
     */
    public boolean exists(final String key){
        return redisTemplate.hasKey(key);
    }


    /**
     * 读取缓存
     * @param key
     * @return
     */
    public Object get(final String key){
        Object result=null;
        ValueOperations<Serializable,Object> operations=redisTemplate.opsForValue();
        result=operations.get(key);
        return result;
    }


    /**
     * 写入缓存
     * @param key
     * @param value
     * @return
     */
    public boolean set(final String key,Object value){
        boolean result=false;
        try{
            ValueOperations<Serializable,Object> operations=redisTemplate.opsForValue();
            operations.set(key,value);
            result=true;
        }catch (Exception e){
            //log.error("RedisUtil set exeption",e);
        }
        return result;
    }


    /**
     * 写入缓存
     * @param key
     * @param value
     * @param expireTime
     * @return
     */
    public boolean set(final String key,Object value,long expireTime){
        boolean result=false;
        try{
            ValueOperations<Serializable,Object> operations=redisTemplate.opsForValue();
            operations.set(key,value);
            redisTemplate.expire(key,expireTime,TimeUnit.SECONDS);
            result=true;
        }catch (Exception e){
            //log.error("RedisUtil set exception",e);
        }
        return result;
    }


    /**
     * 写入缓存
     * 当且仅当key不存在,若给定的key已经存在,则不做任何动作
     * @param key
     * @param value
     * @return
     */
    public boolean setIfAbsent(final String key,Object value){
        boolean result=false;
        try{
            ValueOperations<Serializable,Object> operations=redisTemplate.opsForValue();
            result=operations.setIfAbsent(key,value);
        }catch (Exception e){
            //log.error("RedisUtil setIfAbsent exception",e);
        }
        return result;
    }


    /**
     * 写入缓存
     * 当且仅当key不存在,若给定的key已经存在,则不做任何动作
     * @param key
     * @param value     true:设置成功; false:设置失败,已经存在
     * @param expireTime
     * @return
     */
    public boolean setIfAbsent(final String key,Object value,long expireTime){
        boolean result=false;
        try{
            ValueOperations<Serializable,Object> operations=redisTemplate.opsForValue();
            result=operations.setIfAbsent(key,value);
            if (result){
                redisTemplate.expire(key,expireTime,TimeUnit.SECONDS);
            }
        }catch (Exception e){
            //log.error("RedisUtil setIfAbsent exception",e);
        }
        return result;
    }


    /**
     * 利用指定key作为Redis更新锁(锁有过期时长)
     * @param key
     * @param cacheSeconds
     * @return   获得锁则返回true,否则返回false
     */
    public boolean tryLock(String key,long cacheSeconds){
        boolean isLock=false;
        try {
            isLock=setIfAbsent(key,"",cacheSeconds);
        }catch (Exception e){
            //log.error("RedisUtil tryLock exception",e);
        }
        return isLock;
    }
}

 

 

下面是Redis获取锁和删除锁的一对操作,因为Redis的单线程属性以及setIfAbsent方法,常用于商品秒杀的场景。用syn注解虽然可以解决同步的问题,但是此操作是每次让一个线程进入方法,势必会导致程序的运行的时间过长,不能满足实际的需求,这里推荐使用Redis分布式锁。日常使用中,按照先上锁,运行代码,解锁的步骤运行。

/**
 * @Author Rhine
 * @Date 2018/11/8 2:37
 **/
@Component
@Slf4j
public class RedisLock {

    @Autowired
    private StringRedisTemplate redisTemplate;


    /**
     * 功能描述: 加锁
     * @param: key value 当前时间+超时时间
     * @return: boolean
     * @auther: Rhine
     * @date: 2018/11/8 2:40
     */
    public boolean lock(String key,String value){
        if (redisTemplate.opsForValue().setIfAbsent(key, value)){
            return true;
        }

        String currentValue=redisTemplate.opsForValue().get(key);
        //如果锁过期
        if (!StringUtils.isEmpty(currentValue)&&Long.parseLong(currentValue)<System.currentTimeMillis()){
            //获取上一个锁的时间
            String oldValue=redisTemplate.opsForValue().getAndSet(key, value);
            if (!StringUtils.isEmpty(oldValue)&&oldValue.equals(currentValue)){
                return true;
            }
        }
        return false;
    }


    /**
     * 功能描述: 解锁
     * @param: [key, value]
     * @return: void
     * @auther: Rhine
     * @date: 2018/11/8 3:03
     */
    public void unlock(String key,String value){
        try{
            String currentValue=redisTemplate.opsForValue().get(key);
            if (!StringUtils.isEmpty(currentValue)&&currentValue.equals(value)){
                redisTemplate.opsForValue().getOperations().delete(key);
            }
        }catch (Exception e){
            log.error("【redis分布式锁】 解锁异常,{}",e);
        }
    }
}

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好!对于高性能的 Redis 工具,可以考虑以下几个方面进行设计和实现: 1. 使用 Redis 的原子操作:Redis 提供了多个原子操作指令,如 SETNX(set if not exists)和 EXPIRE(设置过期时间),可以用来实现分布式。可以使用 SETNX 指令尝试获取,如果获取成功则设置过期时间,并在释放使用 DEL 指令删除。 2. 考虑的可重入性:在分布式环境下,同一个线程可能多次请求同一个。为了保证可重入性,可以在的数据结构中增加一个计数器,每次获取时将计数器加一,释放时将计数器减一,只有计数器归零才真正释放。 3. 考虑的可靠性:由于网络等原因,可能存在获取成功但无法设置过期时间的情况。为了避免这种情况导致死,可以使用 Lua 脚本将 SETNX 和 EXPIRE 操作合并为一个原子操作。 4. 考虑的超时机制:为了避免被长时间持有而导致其它线程无法获取,可以为设置一个超时时间。在获取时可以记录当前时间戳,并在释放时检查是否已超过超时时间,如果超时则自动释放。 5. 考虑的释放机制:为了保证的正常释放,可以使用 try-finally 或者 try-with-resources 等机制,在代码块结束时自动释放。 综上所述,您可以基于上述思路设计一个高性能的 Redis 工具,满足分布式环境下的并发控制需求。希望对您有所帮助!如有更多问题,请继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值