1. 程式人生 > >讀寫鎖ReentrantReadWriteLock:讀讀共享,讀寫互斥,寫寫互斥

讀寫鎖ReentrantReadWriteLock:讀讀共享,讀寫互斥,寫寫互斥

  JDK1.5之後,提供了讀寫鎖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時,結果如下:

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

以上例項說明,讀寫鎖ReentrantReadWriteLock:讀讀共享,讀寫互斥,寫寫互斥!