java分布式锁的三种实现方式

分布式锁的核心思想,就是使用外部的一块共享的区域,来完成锁的实现。

一、使用mysql数据库实现(基本不用)

1、使用数据库悲观锁

可以使用select ... for update 来实现分布式锁。

例如:建一个lock表,获取锁就是插入一条数据,移除锁就是删除掉这条数据,使用mysql的for update来保证原子性。

2、使用数据库乐观锁

增加一个version字段,每次更新修改,都会自增加一。

例如:为id是1的用户余额加10

select version,balance from account where user_id ='1';

进行更新时,where条件附带上版本号:

update account set balance = balance+10 ,version = version+1 where version 
= #{version} and user_id ='1';

如果更新失败,则循环上面两步

二、使用redis实现(常用)

1、使用setnx+expire命令实现

setnx是set if not exists 的缩写,也就是只有不存在的时候才设置rediskey 设置成功时返回 1 , 设置失败时返回 0 

if(jedis.setnx(key,lock_value) == 1){ //setnx加锁
    expire(key,100); //设置过期时间
    try {
        do something  //业务处理
    }catch(){
    }
  finally {
       jedis.del(key); //释放锁
    }
}

 缺点:

  • 加锁操作和设置超时时间是分开的。假设在执行完setnx加锁后,正要执行expire设置过期时间时,出现问题,则锁永远无法释放。所以需要使用lua脚本来使setnx+expire成为原子操作
  • 如果超时时间过短,会出现业务还未结束,锁就被释放的情况。

2、Redisson框架(常用)

Redisson是一个知名的、优秀的redis客户端类库,封装了大量的基于redis的复杂的一些操作

redission原理图:

当线程加锁成功后,会启动一个后台线程,会每隔10秒检查一下,还持有锁,那么就会不断的延长锁key的30秒生存时间。因此,Redisson就是使用watch dog解决了锁过期释放,业务没执行完问题

Redisson底层大量使用lua脚本来保证原子性操作,如下面的尝试加锁代码:

简单使用方法:

1、引入依赖

<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
</dependency>

2、配置redisson 

     @Bean(name = "redisClient")
    public RedissonClient redisClient() {
        Config config = new Config();
        config.useSingleServer()
                .setPassword("123456")
                .setTimeout(1000000)
                .setAddress("redis://127.0.0.1:6379");
        ;
        return  Redisson.create(config);
    } 

3、业务中使用 

//1、获取锁
RLock rLock = redisClient.getLock("lock_key");
try{
    //2、加锁
    rLock.lock();
    //业务代码……
}finally{
    //3、释放锁
    rLock.unlock();
}

3、RedLock

redis主从复制架构的问题:

  1. 客户端 A 从 master 获取到锁
  2. 在 master 将锁同步到 slave 之前,master 宕掉了。
  3. slave 节点被晋级为 master 节点
  4. 客户端 B 取得了同一个资源被客户端 A 已经获取到的另外一个锁。安全失效!

正式为了解决上面的问题,才有了基于redis实现的分布式锁——RedLock

它能够保证以下特性:

  • 互斥性:在任何时候,只能有一个客户端能够持有锁;
  • 避免死锁:当客户端拿到锁后,即使发生了网络分区或者客户端宕机,也不会发生死锁;(利用key的存活时间)
  • 容错性:只要多数节点的redis实例正常运行,就能够对外提供服务,加锁或者释放锁;

而非redLock是无法满足互斥性的。

三、使用zookeeper实现

Zookeeper的节点Znode有四种类型:

  • 持久节点:默认的节点类型。创建节点的客户端与zookeeper断开连接后,该节点依旧存在。

  • 持久顺序节点:所谓顺序节点,就是在创建节点时,Zookeeper根据创建的时间顺序给该节点名称进行编号,持久节点顺序节点就是有顺序的持久节点。

  • 临时节点:和持久节点相反,当创建节点的客户端与zookeeper断开连接后,临时节点会被删除。

  • 临时顺序节点:有顺序的临时节点(用于实现分布式锁)

zookeeper分布式锁原理步骤:

  1. zookeeper首先创建一个/lock节点
  2. 当有节点获取锁时,先为这个节点创建临时节点,例如lock-702564158761685-000001,序列号按创建顺序递增。
  3. zookeeper会检查 lock-702564158761685-000001 是否/lock下的最小节点,如果是该节点得到锁,否则监听 lock-702564158761685-000001 前一个节点状态
  4. 当前一个节点的释放锁时会通知(唤醒)后一个节点,(这样只监听前一个节点的方式,避免了线程“惊群效应”)。

简单使用方法:

1、引入依赖

        <dependency>
            <groupId>org.apache.curator</groupId>
            <artifactId>curator-recipes</artifactId>
        </dependency>

 2、创建zk客户端CuratorFramework

CuratorFramework client = CuratorFrameworkFactory.builder()
                    .connectString(CONNECT_ADDR)
                    .retryPolicy(retryPolicy)
                    .connectionTimeoutMs(CONNECTION_TIMEOUT)
                    .sessionTimeoutMs(SESSION_TIMEOUT)
                    .build();
client.start();

 3、业务中使用 


//1、创建临时顺序节点
InterProcessMutex mutex = new InterProcessMutex(client, "/locks/" + workNumber) 
try{
    //2、加锁
    mutex.acquire();
    //业务代码
} finally {
    //3、解锁
    mutex.release();
}

扩展:curator recipes 中的各种锁:

  • InterProcessMutex:可重入、独占锁
  • InterProcessSemaphoreMutex:不可重入、独占锁
  • InterProcessReadWriteLock:读写锁
  • InterProcessSemaphoreV2 : 共享信号量
  • InterProcessMultiLock:多重共享锁 (将多个锁作为单个实体管理的容器)

  • 7
    点赞
  • 40
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java分布式实现方式有多种,常见的包括: 1. 基于Redis的分布式:利用Redis单线程的特性,使用SETNX命令创建,利用EXPIRE设置的过期时间,同时使用DEL命令释放,确保的释放是原子的。 2. 基于Zookeeper的分布式:通过创建临时节点实现分布式,当某个服务占用了,其它服务将无法创建同名节点,从而保证同一时间只有一个服务占用该。 3. 基于数据库的分布式:使用数据库表中的一行记录来表示状态,使用事务确保的获取和释放是原子的。 4. 基于Redisson的分布式:Redisson是一个开源的Java分布式框架,提供了对分布式的支持,使用SETNX和EXPIRE命令实现的创建和过期,同时还提供了自旋、可重入等高级特性。 以上是Java分布式实现方式的几种常见方式,不同的实现方式有着各自的特点和适用场景,需要根据实际需求进行选择。 ### 回答2: Java分布式分布式系统中实现数据同步和控制的关键技术之一,它用于保证多个分布式进程并发访问共享资源时的数据一致性和安全性。分布式与普通的相比,需要解决跨进程、跨节点的同步和并发控制问题。 Java分布式实现方式有以下几种: 1. 基于Zookeeper实现分布式 Zookeeper是一个高性能的分布式协调服务,它可以被用来实现分布式。Zookeeper的实现原理是基于它的强一致性和顺序性,可以保证多个进程访问同一个分布式时的数据同步和控制。 通过创建一个Zookeeper的持久节点来实现分布式,使用create()方法来创建节点,如果创建成功则说明获取成功。当多个进程同时请求获取时,只有一个进程能够创建节点成功,其它进程只能等待。当持有分布式的进程退出时,Zookeeper会自动删除对应的节点,其它进程就可以继续请求获取。 2. 基于Redis实现分布式 Redis是高性能的内存数据库,可以使用它的setnx()命令来实现分布式。setnx()命令可以在指定的key不存在时设置key的值,并返回1;如果key已经存在,则返回0。通过这个原子性的操作来实现分布式。 当多个进程同时请求获取时,只有一个进程能够成功执行setnx()命令,其它进程只能等待。进程在持有期间,可以利用Redis的expire()命令来更新的过期时间。当持有分布式的进程退出时,可以通过delete()命令来删除。 3. 基于数据库实现分布式 数据库通过ACID特性来保证数据的一致性、并发性和可靠性,可以通过在数据库中创建一个唯一索引来实现分布式。当多个进程同时请求获取时,只有一个进程能够成功插入唯一索引,其它进程只能等待。当持有分布式的进程退出时,可以通过删除索引中对应的记录来释放。 不同的实现方式各有优劣。基于Zookeeper的实现方式可以保证分布式的一致性和可靠性,但是需要引入额外的依赖;基于Redis可以实现较高性能的分布式,但是在高并发条件下可能会存在死等问题;基于数据库的实现方式简单,但在高并发条件下也可能会有争抢等问题。 总之,在选择分布式实现方式时,需要根据业务场景和需求来综合考虑各种因素,选择最适合自己的方式。 ### 回答3: 分布式系统中的并发控制是解决分布式系统中竞争资源的重要问题之一,而分布式作为一种并发控制工具,在分布式系统中被广泛采用。Java作为一种常用的编程语言,在分布式实现方面也提供了多种解决方案。下面就分别介绍Java分布式实现方式。 1. 基于ZooKeeper的分布式 ZooKeeper是分布式系统中常用的协调工具,其提供了一套完整的API用于实现分布式实现分布式的过程中需要创建一个Znode,表示,同时用于控制数据的访问。在这个Znode上注册监听器用于接收释放的成功/失败事件,从而控制加/解的过程。 2. 基于Redis的分布式 Redis作为一种高性能的Key-Value数据库,其提供了完整的API用于实现分布式实现分布式的过程中需要在Redis中创建一个Key,利用Redis的SETNX命令进行加,同时设置过期时间保证的生命周期。在解时需要判断是否持有并删除对应的Key。 3. 基于数据库的分布式 数据库作为分布式系统中常用的数据存储方式,其提供了事务机制用于实现分布式。在实现分布式的过程中需要在数据库中创建一个表,利用数据库的事务机制实现/解,同时需要设置过期时间保证的生命周期。 总之,以上三种方式都是常用的Java分布式实现方式。选择合适的方法需要综合考虑的使用场景、性能需求、可靠性要求等因素。同时,在实现分布式的过程中需要注意的加/解的正确性和过期时间的设置,保证分布式系统的并发控制的正确性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值