可重入鎖介紹、使用


1、什么是可重入鎖?

可重入鎖,也叫做遞歸鎖,指的是同一線程外層函數獲得鎖之后 ,內層遞歸函數仍然可以獲取該鎖的代碼,但不受影響。

可重入鎖使用示例

public class SynchronizedSample implements Runnable {
    public synchronized void funA() { // 執行funA方法,需要獲取對象鎖
        // synchronized屬於可重入鎖,進入funA方法后,擁有了對象鎖,所以執行方法funB方法時不受影響。
        System.out.println(Thread.currentThread().getName() + " " + Thread.currentThread().getId());
        funB();
    }
    public synchronized void funB() { // 執行funB方法,需要獲取對象鎖
        System.out.println(Thread.currentThread().getName() + " " + Thread.currentThread().getId());
    }
    @Override
    public void run() {
        funA();
    }
    public static void main(String[] args) {
        SynchronizedSample ss = new SynchronizedSample();
        new Thread(ss).start();
        new Thread(ss).start();
        new Thread(ss).start();
    }
}
public class ReentrantLockSample implements Runnable {
    ReentrantLock lock = new ReentrantLock();
    public void funA() { // 執行funA方法,需要獲取對象鎖
        lock.lock(); // 第一次獲取鎖
        System.out.println(Thread.currentThread().getName() + " " + Thread.currentThread().getId());
        funB(); // 該方法會第二次獲取鎖
        lock.unlock();
    }
    public void funB() { // 執行funB方法,需要獲取對象鎖
        lock.lock();
        System.out.println(Thread.currentThread().getName() + " " + Thread.currentThread().getId());
        lock.unlock();
    }
    @Override
    public void run() {
        funA();
    }
    public static void main(String[] args) {
        ReentrantLockSample rs = new ReentrantLockSample();
        new Thread(rs).start();
        new Thread(rs).start();
        new Thread(rs).start();
    }
}

程序輸出結果

Thread-1 14
Thread-1 14
Thread-2 15
Thread-2 15
Thread-0 13
Thread-0 13

2、可重入鎖作用及使用場景?

  1. 可重入鎖最大作用是避免死鎖。
  2. 當一個線程執行一個帶鎖的代碼塊或方法,同時代碼塊或方法里也獲取同一個鎖。為了避免死鎖,此時就可以用可重入鎖。

3、在內置鎖sychronized和ReentrantLock類之間進行選擇

相同點

  1. 兩種方式在加鎖和內存上提供的語義相同。

不同點

  1. ReentrantLock提供了其他的功能,包括定時的鎖等待、可中斷的鎖等待、公平與非公平鎖。而sychronized沒有這些功能。
  2. ReentrantLock的性能比sychronized要高一些。
  3. sychronized鎖使用方式簡單,代碼表達直接緊湊,使用范圍廣,被許多開發人員所熟悉。
  4. sychronized在線程轉儲中能給出在哪些調用幀中獲得了哪些鎖,並能夠檢測和識別發生死鎖的線程。而java6之后ReentrantLock提供了管理和調試的接口彌補了這一點。
  5. 未來的情況下,可能會提升sychronized而不是ReentrantLock的性能。因為sychronized是JVM的內置屬性,優化可能性很大。

使用建議

在一些內置鎖sychronized無法滿足需求的情況下,ReentrantLock可作為高級工具使用。否則,應該優先考慮使用sychronized。

/**
 * 歡迎評論、留言、發表看法。謝謝!
 */


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM