学习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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 黑盒测试方法的选择策略_选择策略

    黑盒测试方法的选择策略_选择策略

    2022年10月3日
    0
  • 12.Java- Maven 教程

    12.Java- Maven 教程一、安装引用菜鸟教程:Maven菜鸟教程地址1.下载1.Maven链接Maven下载地址:,点击跳转2.选择版本3.解压到指定地址4.并配置环境变量,引用的菜鸟教程引用菜鸟教程,菜鸟教程路径地址,点击跳转右键“计算机”,选择“属性”,之后点击“高级系统设置”,点击”环境变量”,来设置环境变量,有以下系统变量需要配置:新建系统变量MAVEN_HOME,变量值:E:\Maven\apache-maven-3.3.9编辑系统变量Path,添加变量值:;%M

    2022年9月8日
    0
  • 阿里云ddns过程记录

    阿里云ddns过程记录申请了阿里云一年的动态域名,收费的,闲置了几个月,用openwrt一直没成功,最近研究asterisk部署,有个公网的ddns还是方便不少,所以把闲置的域名得拾起来了,过程如下1.开启阿里云后台权限(在访问控制菜单中,文章最后有链接)AliyunDNSReadOnlyAccessAliyunDNSFullAccess2.下载脚本运行GitHub-risfeng/aliyun-ddns-shell:阿里云域名解析动态更新IPShell脚本阿里云域名解析动态更新IPSh

    2022年6月6日
    29
  • 大学项目总结(一)——基于Android的智能点餐系统

    大学项目总结(一)——基于Android的智能点餐系统这个项目是在我大一下学期开始进行制作,并一直到大一暑假结束完成。一共由我和另一个同学两个人完成,我主要负责前端功能,他主要负责后台下面是主要的图册:…

    2022年6月19日
    27
  • QT布局:QVBoxLayout/QHBoxLayout[通俗易懂]

    QT布局:QVBoxLayout/QHBoxLayout[通俗易懂]关系图垂直布局:QVBoxLayout 先来一两代码:QWidget*widget=newQWidget();QVBoxLayout*vBoxLayout=newQVBoxLayout();QLabel*label=newQLabel(“hello”);QPushButton*pushButton=newQPushButton(“hi”);vBoxLayout->addWidget(label);vBoxLayout->addWid…

    2022年6月21日
    37
  • shader 4 杂 一些和函数名词、数据结构

    shader 4 杂 一些和函数名词、数据结构

    2021年11月14日
    39

发表回复

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

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