ReadWriteLock讀寫之間互斥嗎


開發中遇到並發的問題一般會用到鎖,Synchronized存在明顯的一個性能問題就是讀與讀之間互斥;ReadWriteLock是JDK5中提供的讀寫分離鎖。讀寫分離鎖可以有效地幫助減少鎖競爭,以提升系統的性能。

ReadWriteLock管理一組鎖,一個是只讀的鎖,一個是寫鎖。
Java並發庫中ReetrantReadWriteLock實現了ReadWriteLock接口並添加了可重入的特性。

讀寫鎖ReentrantReadWriteLock:讀讀共享,讀寫互斥,寫寫互斥;讀寫鎖維護了一對鎖,一個讀鎖,一個寫鎖,通過分離讀鎖和寫鎖,使得並發性相比一般的排他鎖有了很大提升。在讀多寫少的情況下,讀寫鎖能夠提供比排他鎖更好的並發性和吞吐量。

 

從源碼中可以看出,讀寫鎖中同樣依賴隊列同步器Sync(AQS)實現同步功能,而讀寫狀態就是其同步器的同步狀態。下面從例子中來說明:讀讀共享,讀寫互斥,寫寫互斥

代碼如下:

public class ReentrantWriteReadLockTest {
    ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
    ReadLock readLock = lock.readLock();
    WriteLock writeLock = lock.writeLock();
    
    public void read(){
        try {
            readLock.lock();
            System.out.println("線程"+Thread.currentThread().getName()+"進入。。。");
            Thread.sleep(3000);
            System.out.println("線程"+Thread.currentThread().getName()+"退出。。。");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally{
            readLock.unlock();
        }
    }
    
    public void write(){
        try {
            writeLock.lock();
            System.out.println("線程"+Thread.currentThread().getName()+"進入。。。");
            Thread.sleep(3000);
            System.out.println("線程"+Thread.currentThread().getName()+"退出。。。");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally{
            writeLock.unlock();
        }
    }
    
 
    public static void main(String[] args) {
        final ReentrantWriteReadLockTest wr = new ReentrantWriteReadLockTest();
        Thread t1 = new Thread(new Runnable() {
            public void run() {
                wr.read();
            }
        }, "t1");
        Thread t2 = new Thread(new Runnable() {
            public void run() {
                wr.read();
            }
        }, "t2");
        Thread t3 = new Thread(new Runnable() {
            public void run() {
                wr.write();
            }
        }, "t3");
        Thread t4 = new Thread(new Runnable() {
            public void run() {
                wr.write();
            }
        }, "t4");
        
        t1.start();
        t2.start();
        //t3.start();
        //t4.start();
    }
}

當我們啟動線程t1和t2時,結果如下:

 

線程t1和t2可以同時進入,說明了讀讀共享!

當我們啟動線程t2和t3時,結果如下:

 

一個線程必須等待另一個線程退出,才能進入,說明了讀寫互斥!

當我們啟動線程t3和t4時,結果如下:

 

一個線程必須等待另一個線程退出,才能進入,說明了寫寫互斥!


免責聲明!

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



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