>데이터 베이스 >Redis >Redis와 Java를 사용하여 분산 잠금 기능을 개발하는 방법

Redis와 Java를 사용하여 분산 잠금 기능을 개발하는 방법

WBOY
WBOY원래의
2023-09-21 08:40:461266검색

Redis와 Java를 사용하여 분산 잠금 기능을 개발하는 방법

Redis와 Java를 사용하여 분산 잠금 기능을 개발하는 방법

  1. 소개
    분산 잠금은 분산 시스템에서 공유 리소스에 대한 상호 배타적 액세스를 달성하기 위한 메커니즘입니다. 여러 노드가 동시에 공유 리소스에 액세스하는 경우 한 노드만 액세스하고 다른 노드는 기다려야 하는지 확인해야 합니다. Redis는 일반적으로 사용되는 고성능 및 높은 신뢰성을 갖춘 인메모리 데이터베이스로 분산 잠금을 구현하는 데 매우 적합합니다.
  2. Redis의 setnx 명령
    Redis의 setnx 명령을 사용하여 키 값을 설정할 수 있지만, 키가 존재하지 않는 경우에만 설정 작업이 수행됩니다. 이 기능은 분산 잠금 획득 작업을 구현하는 데 사용할 수 있습니다. setnx 명령을 사용하여 먼저 만료 시간이 있는 키 설정을 시도하십시오. 설정이 성공하면 잠금이 성공적으로 획득되었음을 의미합니다. 그렇지 않으면 다른 노드에서 잠금을 획득했음을 의미합니다.
  3. Java 코드 예시
    다음은 Java 언어와 Redis를 사용하여 분산 잠금을 구현하는 예시 코드입니다.
import redis.clients.jedis.Jedis;

public class DistributedLock {
    private static final String LOCK_KEY = "distributed_lock";
    private static final int LOCK_TIMEOUT = 3 * 1000; // 锁的超时时间,单位为毫秒
    
    private Jedis jedis;

    public DistributedLock(Jedis jedis) {
        this.jedis = jedis;
    }

    public boolean lock() {
        long start = System.currentTimeMillis();
        try {
            while (true) {
                String result = jedis.set(LOCK_KEY, "locked", "NX", "PX", LOCK_TIMEOUT);
                if ("OK".equals(result)) {
                    return true;
                } else {
                    // 进行重试
                    Thread.sleep(100);
                }
                long end = System.currentTimeMillis();
                if (end - start > LOCK_TIMEOUT) {
                    // 超时退出
                    return false;
                }
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            return false;
        }
    }

    public void unlock() {
        jedis.del(LOCK_KEY);
    }
}
  1. 예제 설명
    위 예시 코드에서는 Jedis 라이브러리를 사용하여 Redis를 운영합니다. 먼저, 상수 LOCK_KEY는 분산 잠금의 키로 정의됩니다. 이 키는 모든 노드에서 고유해야 합니다. 또한 잠금 시간 초과를 나타내기 위해 LOCK_TIMEOUT 상수가 설정됩니다. LOCK_KEY作为分布式锁的键,这个键在所有节点中必须保持唯一。另外,设置了一个LOCK_TIMEOUT常量来表示锁的超时时间。

lock方法中,首先获取当前时间作为开始时间,然后使用一个无限循环来尝试获取分布式锁。在循环中,使用Redis的set命令进行设置操作,设置键为LOCK_KEY,值为"locked",并且设置了NXPX选项,NX表示只有键不存在时才执行设置操作,PX表示设置键的过期时间为LOCK_TIMEOUT毫秒。

如果设置成功,则表示获取锁成功,方法返回true;否则继续进行重试,每次重试时会等待100毫秒。同时,还需要判断获取锁的时间是否超过了LOCK_TIMEOUT的值,如果超过则表示获取锁的等待时间已经过长,放弃获取锁,并返回false

unlock方法中,通过调用del

    lock 메서드에서는 먼저 현재 시간을 시작 시간으로 가져온 다음 무한 루프를 사용하여 분산 잠금을 획득하려고 시도합니다. 루프에서 Redis의 set 명령을 사용하여 설정 작업을 수행합니다. 설정 키는 LOCK_KEY이고 값은 "잠김"이며 NX입니다. PX 옵션, NX는 키가 존재하지 않을 때만 설정 작업이 수행됨을 의미하고, PX는 세트의 만료 시간을 의미합니다. 키는 LOCK_TIMEOUT code>밀리초입니다. <li> <br>설정이 성공하면 잠금이 성공적으로 획득되었음을 의미하며 메서드는 <code>true를 반환합니다. 그렇지 않으면 계속 재시도하고 각 재시도마다 100밀리초를 기다립니다. 동시에, 잠금을 획득하는 데 걸리는 시간이 LOCK_TIMEOUT 값을 초과하는지 여부도 확인해야 합니다. 이 값을 초과하면 잠금을 획득하기까지의 대기 시간이 너무 길었다는 의미이며, 잠금 획득을 포기하고 false 를 반환합니다.
unlock 메소드에서 del 명령을 호출하여 분산 잠금의 키를 삭제합니다.

    호출 예시
  1. 다음은 샘플 코드를 이용한 호출 예시입니다.
  2. import redis.clients.jedis.Jedis;
    
    public class LockTest {
        public static void main(String[] args) {
            Jedis jedis = new Jedis("localhost");
            DistributedLock lock = new DistributedLock(jedis);
            try {
                if (lock.lock()) {
                    // 获取到分布式锁后执行需要保护的代码
                    System.out.println("获取到分布式锁");
                    // ... 执行需要保护的代码
                } else {
                    System.out.println("获取分布式锁失败");
                }
            } finally {
                lock.unlock();
            }
        }
    }
호출 예시에서는 먼저 Jedis 연결 객체가 생성된 후 DistributedLock 객체가 생성되고 다음과 같이 Jedis 연결 객체가 전달됩니다. 매개변수. try-finally 블록에서는 먼저 분산 잠금 획득을 시도하고, 성공하면 "분산 잠금 획득"을 출력하고 보호해야 할 코드를 실행한 후 finally 블록에서 분산 잠금을 해제합니다. 🎜🎜🎜요약🎜Redis와 Java 개발을 이용하여 분산 잠금 기능을 쉽게 구현할 수 있습니다. 잠금 획득 작업은 Redis setnx 명령을 사용하여 구현할 수 있으며 Java 코드는 Redis 명령을 쉽게 호출하고 이를 분산 잠금 클래스로 캡슐화할 수 있습니다. 실제 애플리케이션에서는 필요에 따라 분산 잠금의 시간 초과를 조정하여 잠금을 획득하기 위한 대기 시간이 너무 길지 않도록 함으로써 시스템의 성능과 동시성을 향상시킬 수 있습니다. 🎜🎜

위 내용은 Redis와 Java를 사용하여 분산 잠금 기능을 개발하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.