美文网首页
基于Redis的分布式锁

基于Redis的分布式锁

作者: 阿福德 | 来源:发表于2018-09-30 11:42 被阅读0次
/**
     * 阻塞的分布式锁
     * @param lockName
     * @param acquireTimeout
     * @param lockTimeout
     * @param callback
     */
    public static <T> T executeWithRedisLock(String lockName, long acquireTimeout, long lockTimeout, Callback<T> callback) {
        String lockValue="";
        try{
            lockValue = acquireLock(lockName, acquireTimeout, lockTimeout);
            if(lockValue == null) {
                return callback.acquireLockTimeout();
            }
            return callback.execute();
        }finally {
            releaseLock(lockName, lockValue);
        }
    }

    public interface Callback<T>{
        /**
         * execute biz when acquire lock success.
         * @return
         */
        T execute();

        /**
         * execute biz when acquire lock time out.
         * @return
         */
        T acquireLockTimeout();
    }

    /**
     * 阻塞的分布式锁 pair with acquireLock
     * @param lockName 锁名称,redisKey
     * @param acquireTimeout 获取锁的超时时间,单位毫秒
     * @param lockTimeout 锁超时时间,单位毫秒
     * @return 得到锁的值,null为获取锁超时
     */
    private static String acquireLock(String lockName, long acquireTimeout, long lockTimeout) {
        Jedis conn = getJedisPool().getResource();
        boolean broken = false;
        try {
            String identifier = System.nanoTime() + "";   //锁的值
            String lockKey = DISTRIBUTE_LOCK_PREFIX + lockName;     //锁的键
            int lockExpire = (int) (lockTimeout / 1000);     //锁的过期时间

            long end = System.currentTimeMillis() + acquireTimeout;     //尝试获取锁的时限
            while (System.currentTimeMillis() < end) {      //判断是否超过获取锁的时限
                if (conn.setnx(lockKey, identifier) == 1) {  //判断设置锁的值是否成功
                    conn.expire(lockKey, lockExpire);   //设置锁的过期时间
                    return identifier;          //返回锁的值
                }

                try {
                    Thread.sleep(20);    //等待1秒后重新尝试设置锁的值
                } catch (InterruptedException ie) {
                    Thread.currentThread().interrupt();
                }
            }
            String value = conn.get(lockKey);
            if(value != null) {
                if(System.nanoTime() - Long.valueOf(value) > (lockTimeout + REDIS_LOCK_TOLERATION_TIMEOUT) * 1000000) {
                    conn.del(lockKey);
                }
            }

            // 获取锁失败时返回null
            return null;
        }catch (JedisException e) {
            broken = handleJedisException(e);
            return null;
        }finally {
            closeResource(conn, broken);
        }
    }

    /**
     * 释放分布式锁, pair with acquireLock
     * @param lockName 锁名称,redisKey
     * @return 释放锁释是否成功
     */
    private static void releaseLock(String lockName, String identifier) {
        Jedis conn = null;
        boolean broken = false;
        try {
            conn = getJedisPool().getResource();
            String lockKey = DISTRIBUTE_LOCK_PREFIX + lockName;
            if (identifier != null && identifier.equals(conn.get(lockKey))) {
                conn.del(lockKey);
            }
        }catch (JedisException e) {
            broken = handleJedisException(e);
        }finally {
            if(conn!=null) {
                closeResource(conn, broken);
            }
        }
    }
    private static boolean handleJedisException(JedisException jedisException) {
        if (jedisException instanceof JedisDataException) {
            if ((jedisException.getMessage() == null) || (jedisException.getMessage().indexOf("READONLY") == -1)) {
                return false;
            }
        }
        return true;
    }

    private static void closeResource(Jedis jedis, boolean conectionBroken) {
        if (conectionBroken) {
            jedisPool.returnBrokenResource(jedis);
        } else {
            jedisPool.returnResource(jedis);
        }
    }

相关文章

  • 基于redis实现的分布式锁

    本文要点 基于redis实现分布式锁demo 基于redis实现分布式锁原理 基于redis实现分布式锁优缺点 正...

  • 分布式锁实现

    基于数据库实现分布式锁基于缓存(redis,memcached)实现分布式锁基于Zookeeper实现分布式锁 s...

  • 7.2-基于Redis实现分布式锁的几种坑你是否踩过《上》—小滴

    基于Redis实现分布式锁的几种坑你是否踩过《上》 简介:基于Redis实现分布式锁的几种坑 实现分布式锁 可以用...

  • 基于redis的分布式锁

    分布式锁实现方案 基于数据库实现分布式锁 基于缓存(redis,memcached,tair)实现分布式锁 基于Z...

  • 分布式锁的实现方式

    分布式锁通常有3种实现方式,即数据库乐观锁、基于redis的分布式锁和基于zookeeper的分布式锁。 一、基于...

  • 分布式锁之Redis实现(acquire)

    分布式锁一般有三种实现方式: 基于数据库的锁; 基于Redis的分布式锁; 基于ZooKeeper的分布式锁。 本...

  • Redis实现分布式锁

    分布式下的分布式锁一般实现有三种: 基于数据库的乐观锁 基于redis的分布式锁 基于zookeeper的分布式锁...

  • 84 redis实现分布式锁的原理

    1,Redis使用setnx 实现2,Redisson 分布式锁;Redis基于 setnx 实现分布式锁原理:R...

  • Redis实现分布式锁

    1. 分布式锁分类 数据库乐观锁 基于Redis的分布式锁 基于ZooKeeper的分布式锁 2. 组件依赖 po...

  • Redis分布式锁实现

    分布式锁实现方式 数据库乐观锁; 基于Redis的分布式锁; 基于ZooKeeper的分布式锁。 本篇将介绍第二种...

网友评论

      本文标题:基于Redis的分布式锁

      本文链接:https://www.haomeiwen.com/subject/rdqwoftx.html