学习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();
        }
    }
}

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

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

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • CS模型下载_利益相关者模型

    CS模型下载_利益相关者模型当前统计模型,CS模型基于当前统计模型的无迹卡尔曼滤波目标基于当前统计模型的容积卡尔曼滤波目标机动目标跟踪——当前统计模型(CS模型)1.对机动目标跟踪的理解1.1.对机动目标跟踪的理解1.2.目标模型概述2.”当前”统计CS模型3.”当前”统计CS模型3.1.”当前”统计CS模型(连续)3.2.”当前”统计CS模型(离散)3.3.”当前”统计CS模型分析4.”当前”统计CS模型(二维)4.1.”当前”统计CS模型(连续)………………

  • CMS-项目的技术架构

    CMS-项目的技术架构2项目的技术架构2.1技术架构学成在线采用当前流行的前后端分离架构开发,由用户层、UI层、微服务层、数据层等部分组成,为PC、App、H5等客户端用户提供服务。下图是系统的技术架构图:业务流程举例:用户可以通过pc、手机等客户端访问系统进行在线学习。系统应用CDN技术,对一些图片、CSS、视频等资源从CDN调度访问。所有的请求全部经过负载均衡器。对于PC、H5等客户端请求,…

  • RTP协议与实战

    RTP协议与实战一.前言二.RTP协议三.使用jrtplib发送RTP数据包

  • jquery实现tab切换完整代码

    代码如下,保存到html文件打开:1234jquery实现tab切换-柯乐义554555657777879808182jQuery83Javascript84

    2021年12月20日
  • 自己工作用到的 linux添加路由的方法

    自己工作用到的 linux添加路由的方法linux下添加路由的方法一、查看及添加临时路由1.查看路由(linux下)[root@nfs~]#route#同netstat-rnKernelIProutingtableDestinationGatewayGenmaskFlagsMetricRefUseIface192.168.90.2*255.255.255.255UH00

  • git切换到指定远程分支

    git切换到指定远程分支我们在使用git进行开发的时候经常会遇到需要切换远程分支并且提交到远程指定分支的情况,现在记录下操作步骤。查看远程所有分支$gitbranch-agitbranch不带参数,列出本地已经存在的分支,并且在当前分支的前面用*标记,加上-a参数可以查看所有分支列表,包括本地和远程,远程分支一般会用红色字体标记出来*devmasterremote…

发表回复

您的电子邮箱地址不会被公开。

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