学习redis实现分布式锁—–自己的一个理解

学习redis实现分布式锁—–自己的一个理解

一、导入maven依赖
<!– 操作redis的java客户端jedis –>
        <dependency>
            <groupId>redis.clients</groupId>
            <artifactId>jedis</artifactId>
            <version>2.9.0</version>
        </dependency>
二、代码实现
import java.util.UUID;

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;

/**
 * 分布式锁的思想:多个服务器集群下,只允许其中一个jvm进行操作 
 * redis实现分布式锁的思路:
 * 
 * 获取锁:在多个jvm情况下,redis实现分布式锁通过setnx方法创建同一个key ,
 *      这个key是唯一不重复,如果存入成功返回1获取锁,存入失败返回0,value使用一个唯一不重复的随机数作为线程的ID。
 * 
 * 释放锁:为了保证线程获取的锁,于释放的锁是同一个,在删除reids的key的时候,要对线程ID进行判断,是同一个ID才进行删除
 * 
 * 如何防止死锁 
 *            1.设置一个获取锁之前的一个时间段,线程如果在这个时间段还没有获取到锁,那么线程就放弃获取锁,返回null
 *         2.还要设置另一个时间段,就是线程获取到锁之后,对key设置有效时间,过了这个是时间段,key自动删除 ,释放锁  返回线程ID
 * 
 * @author zxlovey
 *
 */
public class LockRedis {

    // jedis线程池
    private JedisPool jedisPool;

    public LockRedis(JedisPool jedisPool) {

        this.jedisPool = jedisPool;
    }

    private String reidsKey = “redis_key”;

    // 获取锁
    /**
     * 
     * @param acquireTimeout
     *            获取锁之前的一个时间段
     * @param timeOut
     *            线程获取到锁之后,对key设置有效时间 一般以秒为单位
     */
    public String getLockRedis(Long acquireTimeout, Long timeOut) {

        Jedis conn = null;
        try {

            // 获取到jedis
            conn = jedisPool.getResource();

            Long endTime = System.currentTimeMillis() + acquireTimeout;
            int expireLock = (int) (timeOut / 1000);

            String identifierValue = UUID.randomUUID().toString();

            while (System.currentTimeMillis() < endTime) {

                if (conn.setnx(reidsKey, identifierValue) == 1) {

                    // 获取锁 设置有效时间
                    conn.expire(reidsKey, expireLock);
                    // 返回value,这个value作为线程id 释放锁的时候需要判断删除的key 和获取的锁是同一个
                    return identifierValue;
                }
            }

        } catch (Exception e) {

            // TODO: handle exception
        } finally {

            if (conn != null) {

                conn.close();
            }
        }

        return null;
    }

    // 释放锁
    /**
     * 
     * @param identifierValue
     *            锁的ID,就是key 对应的value
     */
    public void unLockRedis(String identifierValue) {

        Jedis conn = null;
        try {

            conn = jedisPool.getResource();

            if (conn.get(reidsKey).equals(identifierValue)) {

                // 获取的key的value相同,说明是同一个线程的锁资源
                // 删除key
                conn.del(reidsKey);
                System.out.println(“释放锁成功:” + Thread.currentThread().getName() + “—锁的ID:” + identifierValue);
            }

        } catch (Exception e) {

        } finally {

            if (conn != null) {

                conn.close();
            }
        }
    }

}

import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

public class LockServer {

    private static JedisPool pool = null;

    static {

        JedisPoolConfig config = new JedisPoolConfig();
        // 设置最大连接数
        config.setMaxTotal(200);
        // 设置最大空闲数
        config.setMaxIdle(8);
        // 设置最大等待时间
        config.setMaxWaitMillis(1000 * 100);
        // 在borrow一个jedis实例时,是否需要验证,若为true,则所有jedis实例均是可用的
        config.setTestOnBorrow(true);
        pool = new JedisPool(config, “127.0.0.1”, 6379, 3000);
    }
    
    private LockRedis lockRedis = new LockRedis(pool);
    
        //演示redis实现分布式锁
        public void seckill() {

            //1.获取到锁
            String identifierValue = lockRedis.getLockRedis(1000L, 1000L);
            if(identifierValue == null){

                System.out.println(“获取锁失败:”+Thread.currentThread().getName()+”;失败原因是获取锁的时间超时”);
                return;
            }
            System.out.println(“获取锁成功:”+Thread.currentThread().getName()+”—锁的ID:”+identifierValue);
            
            //2.释放锁
            lockRedis.unLockRedis(identifierValue);    
        }
    
}

public class ThreadLock extends Thread{

    private LockServer lockServer;
    
      public ThreadLock(LockServer lockServer) {

        this.lockServer = lockServer;
    }
    
    @Override
    public void run() {

         lockServer.seckill();
    }
    
}

public class Test001 {

    public static void main(String[] args) {

        LockServer lockServer = new LockServer();
        for (int i = 0; i < 500; i++) {

            ThreadLock tl = new ThreadLock(lockServer);
            tl.start();
        }
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/100803.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • docker 镜像构建_如何更新docker镜像内的文件

    docker 镜像构建_如何更新docker镜像内的文件前言如果我们已经安装了一个python3的环境,如果另一台机器也需要安装同样的环境又要敲一遍,很麻烦,这里可以配置Dockerfile文件,让其自动安装,类似shell脚本Dockerfile编写

    2022年7月31日
    7
  • 学计算机的男生喜欢什么样的女生,it男喜欢什么样的女孩

    学计算机的男生喜欢什么样的女生,it男喜欢什么样的女孩喜欢什么样的女孩子每个男生的定义都不一样,有些男生更喜欢性感魅惑的御姐范儿气质,当然,也有很多男生偏向甜美可爱风格的。那么it男喜欢什么样的女孩子呢?来源:图虫作者:79054344888189459虽然很多花心的男生喜欢的女孩子的类型是不固定的,而且一般都是见一个爱一个的,但实际上很多男生的心里对于自己喜欢的女孩子的类型是很清楚的,所以就算有女生喜欢他们,他们也不反感,但是接受却是一件困难的事情…

    2022年7月25日
    19
  • 视频直播技术详解之一:开篇

    视频直播技术详解之一:开篇随着互联网用户消费内容和交互方式的升级,支撑这些内容和交互方式的基础设施也正在悄悄发生变革。手机设备拍摄视频能力和网络的升级催生了大家对视频直播领域的关注,吸引了很多互联网创业者或者成熟企业进入该领域。七牛云作为一家以基础服务能力见长的云计算公司,于6月底发布了一个针对视频直播的实时流网络LiveNet和完整的直播云解决方案,很多开发者对这个网络和解决方案的细节和使用场景非常感兴趣

    2022年7月21日
    14
  • web基础知识_Web开发基础

    web基础知识_Web开发基础传统身份验证的方法:HTTP是一种没有状态的协议,也就是它并不知道是谁是访问应用。这里我们把用户看成是客户端,客户端使用用户名还有密码通过了身份验证,不过下回这个客户端再发送请求时候,还得再验证一下。解决的方法就是,当用户请求登录的时候,如果没有问题,我们在服务端生成一条记录,这个记录里可以说明一下登录的用户是谁,然后把这条记录的ID号发送给客户端,客户端收到以后把这个ID号存储在C…

    2022年9月13日
    0
  • 编写socket套接字的步骤_windows7任务栏组成部分

    编写socket套接字的步骤_windows7任务栏组成部分详细内容请见书Windows程序设计珍藏版1081页或请见文章:https://www.jianshu.com/p/066d99da7cbd

    2022年8月18日
    3
  • python系列文章(基础,应用,后端,运维,自动化测试,爬虫,数据分析,可视化,机器学习,深度学习系列内容)

    python基础教程python基础系列教程——Python的安装与测试:python解释器、PyDev编辑器、pycharm编译器python基础系列教程——Python库的安装与卸载python基础系列教程——Python3.x标准模块库目录python基础系列教程——Python中的编码问题,中文乱码问题python基础系列教程——python基础语法全解python…

    2022年4月18日
    39

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号