Heim  >  Artikel  >  Datenbank  >  Was ist eine Wiedereintrittssperre? Detaillierte Erläuterung, wie Redis verteilte Wiedereintrittssperren implementiert

Was ist eine Wiedereintrittssperre? Detaillierte Erläuterung, wie Redis verteilte Wiedereintrittssperren implementiert

青灯夜游
青灯夜游nach vorne
2022-01-28 10:00:108885Durchsuche

Was ist eine Wiedereintrittssperre? Wie implementiert man eine Wiedereintrittssperre? Der folgende Artikel gibt Ihnen eine ausführliche Diskussion darüber, wie Redis verteilte Wiedereintrittssperren implementiert. Ich hoffe, er wird Ihnen hilfreich sein!

Was ist eine Wiedereintrittssperre? Detaillierte Erläuterung, wie Redis verteilte Wiedereintrittssperren implementiert

Was ist eine nicht wiedereintretende Sperre?

Das heißt, wenn der aktuelle Thread eine Methode ausführt und die Sperre erworben hat, kann er sie beim erneuten Versuch, die Sperre in der Methode zu erlangen, nicht erlangen und wird blockiert.

Was ist eine Wiedereintrittssperre?

Wiedereintretende Sperre, auch rekursive Sperre genannt, bedeutet, dass im selben Thread die innere rekursive Funktion die Sperre weiterhin erhalten kann, nachdem die äußere Funktion die Sperre erhalten hat. Das heißt, wenn derselbe Thread denselben Code erneut eingibt, kann er die Sperre erneut erhalten.

Wiedereinschaltbare Sperrfunktion?

Verhindern Sie, dass ein Deadlock die Sperre mehrmals im selben Thread erhält.

Hinweis: In der Java-Programmierung sind synchronisiert und ReentrantLock beide Wiedereintrittssperren. 在java的编程中synchronized 和 ReentrantLock都是可重入锁。

基于synchronized的可重入锁

步骤1:双重加锁逻辑

public class SynchronizedDemo {
    //模拟库存100
    int count=100;
    public synchronized void operation(){
        log.info("第一层锁:减库存");
        //模拟减库存
        count--;
        add();
        log.info("下订单结束库存剩余:{}",count);
    }

    private synchronized void add(){
        log.info("第二层锁:插入订单");
        try {
            Thread.sleep(1000*10);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

步骤2:加个测试类

public static void main(String[] args) {
    SynchronizedDemo synchronizedDemo=new SynchronizedDemo();
    for (int i = 0; i < 3; i++) {
        int finalI = i;
        new Thread(()->{
            log.info("-------用户{}开始下单--------", finalI);
            synchronizedDemo.operation();
        }).start();
    }
}

步骤3:测试

20:44:04.013 [Thread-2] INFO com.agan.redis.controller.SynchronizedController - -------用户2开始下单--------
20:44:04.013 [Thread-1] INFO com.agan.redis.controller.SynchronizedController - -------用户1开始下单--------
20:44:04.013 [Thread-0] INFO com.agan.redis.controller.SynchronizedController - -------用户0开始下单--------
20:44:04.016 [Thread-2] INFO com.agan.redis.Reentrant.SynchronizedDemo - 第一层锁:减库存
20:44:04.016 [Thread-2] INFO com.agan.redis.Reentrant.SynchronizedDemo - 第二层锁:插入订单
20:44:14.017 [Thread-2] INFO com.agan.redis.Reentrant.SynchronizedDemo - 下订单结束库存剩余:99
20:44:14.017 [Thread-0] INFO com.agan.redis.Reentrant.SynchronizedDemo - 第一层锁:减库存
20:44:14.017 [Thread-0] INFO com.agan.redis.Reentrant.SynchronizedDemo - 第二层锁:插入订单
20:44:24.017 [Thread-0] INFO com.agan.redis.Reentrant.SynchronizedDemo - 下订单结束库存剩余:98
20:44:24.017 [Thread-1] INFO com.agan.redis.Reentrant.SynchronizedDemo - 第一层锁:减库存
20:44:24.017 [Thread-1] INFO com.agan.redis.Reentrant.SynchronizedDemo - 第二层锁:插入订单
20:44:34.017 [Thread-1] INFO com.agan.redis.Reentrant.SynchronizedDemo - 下订单结束库存剩余:97
  • 由于synchronized关键字修饰的是方法,所有加锁为实例对象:synchronizedDemo
  • 运行结果可以看出减库存和插入订单都是每个线程都完整运行两个方法完毕,才能释放锁,其他线程才能拿锁,即是一个线程多次可以拿到同一个锁,可重入。所以synchronized也是可重入锁。

基于ReentrantLock的可重入锁

ReentrantLock,是一个可重入且独占式的锁,是一种递归无阻塞的同步锁。和synchronized关键字相比,它更灵活、更强大,增加了轮询、超时、中断等高级功能。

步骤1:双重加锁逻辑

public class ReentrantLockDemo {

    private Lock lock =  new ReentrantLock();

    public void doSomething(int n){
        try{
            //进入递归第一件事:加锁
            lock.lock();
            log.info("--------递归{}次--------",n);
            if(n<=2){
                try {
                    Thread.sleep(1000*2);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                this.doSomething(++n);
            }else{
                return;
            }
        }finally {
            lock.unlock();
        }
    }

}

步骤2:加个测试类

public static void main(String[] args) {
    ReentrantLockDemo reentrantLockDemo=new ReentrantLockDemo();
    for (int i = 0; i < 3; i++) {
        int finalI = i;
        new Thread(()->{
            log.info("-------用户{}开始下单--------", finalI);
            reentrantLockDemo.doSomething(1);
        }).start();
    }
}

步骤3:测试

20:55:23.533 [Thread-1] INFO com.agan.redis.controller.ReentrantController - -------用户1开始下单--------
20:55:23.533 [Thread-2] INFO com.agan.redis.controller.ReentrantController - -------用户2开始下单--------
20:55:23.533 [Thread-0] INFO com.agan.redis.controller.ReentrantController - -------用户0开始下单--------
20:55:23.536 [Thread-1] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归1次--------
20:55:25.537 [Thread-1] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归2次--------
20:55:27.538 [Thread-1] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归3次--------
20:55:27.538 [Thread-2] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归1次--------
20:55:29.538 [Thread-2] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归2次--------
20:55:31.539 [Thread-2] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归3次--------
20:55:31.539 [Thread-0] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归1次--------
20:55:33.539 [Thread-0] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归2次--------
20:55:35.540 [Thread-0] INFO com.agan.redis.Reentrant.ReentrantLockDemo - --------递归3次--------
  • 运行结果可以看出,每个线程都可以多次加锁解锁的,ReentrantLock是可重入的。

redis如何实现分布式重入锁?

setnx虽然可以实现分布式锁,但是不可重入,在一些复杂的业务场景,我们需要分布式重入锁时, 对于redis的重入锁业界还是有很多解决方案的,目前最流行的就是采用RedissonWiedereintrittssperre basierend auf synchronisiertem

Schritt 1: Doppelte Sperrlogik

public class RedisController {

    @Autowired
    RedissonClient redissonClient;

    @GetMapping(value = "/lock")
    public void get(String key) throws InterruptedException {
        this.getLock(key, 1);
    }

    private void getLock(String key, int n) throws InterruptedException {
        //模拟递归,3次递归后退出
        if (n > 3) {
            return;
        }
        //步骤1:获取一个分布式可重入锁RLock
        //分布式可重入锁RLock :实现了java.util.concurrent.locks.Lock接口,同时还支持自动过期解锁。
        RLock lock = redissonClient.getLock(key);
        //步骤2:尝试拿锁
        // 1. 默认的拿锁
        //lock.tryLock();
        // 2. 支持过期解锁功能,10秒钟以后过期自动解锁, 无需调用unlock方法手动解锁
        //lock.tryLock(10, TimeUnit.SECONDS);
        // 3. 尝试加锁,最多等待3秒,上锁以后10秒后过期自动解锁
        // lock.tryLock(3, 10, TimeUnit.SECONDS);
        boolean bs = lock.tryLock(3, 10, TimeUnit.SECONDS);
        if (bs) {
            try {
                // 业务代码
                log.info("线程{}业务逻辑处理: {},递归{}" ,Thread.currentThread().getName(), key,n);
                //模拟处理业务
                Thread.sleep(1000 * 5);
                //模拟进入递归
                this.getLock(key, ++n);
            } catch (Exception e) {
                log.error(e.getLocalizedMessage());
            } finally {
                //步骤3:解锁
                lock.unlock();
                log.info("线程{}解锁退出",Thread.currentThread().getName());
            }
        } else {
            log.info("线程{}未取得锁",Thread.currentThread().getName());
        }
    }
}

  • Schritt 2: Testklasse hinzufügen
  • 线程http-nio-9090-exec-1业务逻辑处理: ljw,递归1
    线程http-nio-9090-exec-2未取得锁
    线程http-nio-9090-exec-1业务逻辑处理: ljw,递归2
    线程http-nio-9090-exec-3未取得锁
    线程http-nio-9090-exec-1业务逻辑处理: ljw,递归3
    线程http-nio-9090-exec-1解锁退出
    线程http-nio-9090-exec-1解锁退出
    线程http-nio-9090-exec-1解锁退出
  • Schritt 3: Testen
    • rrreee
    • Da das synchronisierte Schlüsselwort Methoden ändert, sind alle Sperren Instanzobjekte: synchronisiertDemo
    • Die laufenden Ergebnisse zeigen, dass sowohl die Bestandsreduzierung als auch die Auftragseinfügung erfordern, dass jeder Thread die beiden Methoden vollständig ausführt, bevor die Sperre aufgehoben werden kann Threads können die Sperre erhalten, das heißt, ein Thread kann dieselbe Sperre mehrmals erhalten und wiedereintrittsfähig sein. Synchronisiert ist also auch eine Wiedereintrittssperre.
    • Reentrant-Sperre basierend auf ReentrantLock
    • ReentrantLock ist eine wiedereintretende und exklusive Sperre sowie eine rekursive, nicht blockierende Synchronisationssperre. Im Vergleich zum synchronisierten Schlüsselwort ist es flexibler und leistungsfähiger und bietet erweiterte Funktionen wie Polling, Timeout und Interrupt. 🔜 Die Ergebnisse können in jedem Thread gesehen werden kann mehrmals gesperrt und entsperrt werden, und ReentrantLock ist wiedereintrittsfähig.
Wie implementiert Redis verteilte Wiedereintrittssperren?

Obwohl setnx verteilte Sperren implementieren kann, ist es in einigen komplexen Geschäftsszenarien nicht wiedereintrittsfähig, wenn wir verteilte wiedereintrittsfähige Sperren benötigen. Es gibt viele Lösungen für die Wiedereintrittssperre von Redis in der Branche, und die beliebteste ist derzeit die Verwendung von Redisson. [Verwandte Empfehlungen: Redis-Video-Tutorial]

Was ist Redisson?

Redisson ist die von Redis offiziell empfohlene Java-Version des Redis-Clients.

Basierend auf den gemeinsamen Schnittstellen im Java Utility Toolkit bietet es Benutzern eine Reihe häufig verwendeter Toolklassen mit verteilten Eigenschaften.

In Bezug auf die Netzwerkkommunikation basiert es auf dem Netty-Framework von NIO, um eine hohe Leistung der Netzwerkkommunikation sicherzustellen.

In Bezug auf die Funktion verteilter Sperren bietet es eine Reihe verteilter Sperren, z. B.: ReadWriteLock)

Interlock(MultiLock)
  • Red Lock(RedLock)
    • Case Combat: Erleben Sie die verteilte Wiedereintrittssperre von Redis
    1. Schritt 1: Redisson-Konfiguration
    .
    • Redisson Konfiguration kann überprüft werden: Redis Distributed Cache (34) – SpringBoot-Integration mit Redission – Nuggets (juejin.cn)
    https://juejin.cn/post/7057132897819426824
    1. Schritt 2: Redisson-Re-Entry-Sperrtestklasse
    rrreee
      RLock verfügt über drei Sperraktionen:
    Standardsperre

lock.tryLock(); 🎜🎜🎜🎜🎜🎜 unterstützt Ablaufentriegelungsfunktion, 10 Sekunden. Es wird nach Ablauf in der Zukunft automatisch entsperrt 🎜🎜🎜🎜sperre. tryLock(10, TimeUnit.SECONDS);🎜🎜🎜🎜🎜🎜Versuchen Sie zu sperren, warten Sie bis zu 3 Sekunden und es wird nach Ablauf von 10 Sekunden nach dem Sperren automatisch entsperrt🎜🎜🎜🎜🎜lock. tryLock(3, 10, TimeUnit .SECONDS);🎜🎜🎜🎜🎜🎜Unterschied: 🎜
  • lock.lock():阻塞式等待。默认加的锁都是30s
    • 锁的自动续期,如果业务超长,运行期间自动锁上新的30s。不用担心业务时间长而导致锁自动过期被删掉(默认续期)
    • 加锁的业务只要运行完成,就不会给当前锁续期,即使不手动解锁,锁默认会在30s内自动过期,不会产生死锁问题
    • lock()如果我们未指定锁的超时时间,就使用【看门狗默认时间】: lockWatchdogTimeout = 30 * 1000
    • 原理:只要占锁成功,就会启动一个定时任务【重新给锁设置过期时间,新的过期时间就是看门狗的默认时间】,每隔10秒都会自动的再次续期,续成30秒
  • lock.lock(10,TimeUnit.SECONDS) :10秒钟自动解锁,自动解锁时间一定要大于业务执行时间
    • 出现的问题:在锁时间到了以后,不会自动续期
    • 原理:lock(10,TimeUnit.SECONDS)如果我们传递了锁的超时时间,就发送给redis执行脚本,进行占锁,默认超时就是我们制定的时间

最佳实战:

  • lock.lock(10,TimeUnit.SECONDS);  省掉看门狗续期操作,自动解锁时间一定要大于业务执行时间,手动解锁

步骤3:测试

访问3次:http://127.0.0.1:9090/lock?key=ljw

线程http-nio-9090-exec-1业务逻辑处理: ljw,递归1
线程http-nio-9090-exec-2未取得锁
线程http-nio-9090-exec-1业务逻辑处理: ljw,递归2
线程http-nio-9090-exec-3未取得锁
线程http-nio-9090-exec-1业务逻辑处理: ljw,递归3
线程http-nio-9090-exec-1解锁退出
线程http-nio-9090-exec-1解锁退出
线程http-nio-9090-exec-1解锁退出

通过测试结果:

  • nio-9090-exec-1线程,在getLock方法递归了3次,即证明了lock.tryLock是可重入锁
  • 只有当nio-9090-exec-1线程执行完后,io-9090-exec-2 nio-9090-exec-3 未取得锁 因为lock.tryLock(3, 10, TimeUnit.SECONDS),尝试加锁,最多等待3秒,上锁以后10秒后过期自动解锁 所以等了3秒都等不到,就放弃了

总结

上面介绍了分布式重入锁的相关知识,证明了Redisson工具能实现了可重入锁的功能。其实Redisson工具包中还包含了读写锁(ReadWriteLock)和 红锁(RedLock)等相关功能,我们下篇文章再详细研究。

更多编程相关知识,请访问:编程入门!!

Das obige ist der detaillierte Inhalt vonWas ist eine Wiedereintrittssperre? Detaillierte Erläuterung, wie Redis verteilte Wiedereintrittssperren implementiert. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:juejin.cn. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen