Java中Lock框架學習筆記

      鎖在多線程編程中有很重要的做用,synchronized比較常見也很經常使用,可是Lock提供了更普遍的鎖操做,處理多線程同步的問題也更加優雅和靈活,Java從Java SE 5以後在併發包中提供Lock接口。編程

1、Lock和synchronized的區別和各自的特色多線程

一、類型不一樣
Lock是一個接口,是JDK層面的實現;synchronized是Java的關鍵字,是JVM層面的實現,是Java的內置特性;這也形成了在鎖管理上的不一樣。
二、釋放鎖的方式
Lock是在編碼中進行鎖的操做,須要主動調用接口中的方法釋放鎖,因此使用Lock時須要配合try模塊,在finally中釋放鎖,否則一旦發生異常極可能形成死鎖現象;synchronized因爲是JVM層面的實現,獲取鎖和釋放鎖的操做是不可見的,當發生異常時,鎖的釋放由JVM實現。
三、獲取鎖的過程
Lock接口中提供的方法,讓獲取鎖的過程更加靈活,編程人員能夠方便的在獲取鎖的過程當中進行多種操做,好比嘗試獲取鎖、設置獲取鎖的超時時間、中斷等待獲取鎖的線程等,應該說讓鎖的操做變得「豐富多彩」起來;synchronized是沒法這麼靈活的對鎖進行操做的。
四、效率
基於讀寫鎖接口的擴展,Lock能夠提升多個線程進行讀操做的效率,在大併發量的狀況下,效率的提升會尤爲明顯。併發

2、Lock應用場景舉例編碼

一、解決獲取鎖的等待問題
若是佔有鎖的線程A因爲各類緣由致使阻塞而沒有釋放鎖,此時其餘線程B也須要得到該鎖。synchronized的機制是讓B持續等待,若是A一直沒有釋放鎖,那麼B將一直等待,這會很大程度影響執行的效率;而Lock中提供了中斷線程等待的方法,也提供了帶有超時時間的獲取鎖的方法,後面會講到這些方法。
二、讀寫鎖的分離
咱們知道,多線程僅僅是執行讀操做的話是沒有衝突問題的,於是在讀操做時的鎖不必是獨佔的。synchronized實現同步就會致使在讀操做時只能有一個線程得到鎖,其餘線程只能等待鎖的釋放。Lock中的ReentrantReadWriteLock很好的解決了這個問題。
三、其餘鎖的操做
如獲知當前線程是否成功得到鎖等,synchronized是作不到的。spa

3、Lock接口中的方法線程

先看一下Lock的源碼,它是一個接口:接口

 

public interface Lock {資源

    void lock();get

 

    void lockInterruptibly() throws InterruptedException;同步

 

    boolean tryLock();

 

    boolean tryLock(long var1, TimeUnit var3) throws InterruptedException;

 

    void unlock();

 

    Condition newCondition();

}

 

在介紹每一個方法以前,先說明兩個注意事項:
一、Lock的實例通常定義爲成員變量,若是定義爲局部變量,每一個線程都會保存一個本身的副本,那麼在獲取鎖的操做時,實際每一個線程獲取的是不一樣的鎖,沒法造成同步互斥訪問。
二、獲取鎖的操做要放在try模塊以外,緣由是若是放在try模塊內的話,當獲取鎖的操做發生異常會調用finally中的代碼釋放鎖,而此時可能並無獲取鎖,就會拋出異常。
接下來看一下每一個方法的做用:
1、lock()
特色:發生異常不自動釋放鎖;若是沒有獲取到鎖會等待;
代碼示例:

    private Lock lock = new ......; //建立鎖

 

    public void getLock() {

        lock.lock(); //得到鎖

        try {

            System.out.print("業務處理"); //任務處理

        } catch (Exception e) {

 

        } finally {

            lock.unlock(); //釋放鎖

        }

    }

2、tryLock()
特色:帶有boolean型返回值;不管是否成功獲取鎖會當即返回不進行等待;

    private Lock lock = new ......; //建立鎖

 

    public void getLock() {

 

       if(lock.tryLock()) {

           try {

               System.out.print("業務處理"); //任務處理

           } catch (Exception e) {

 

           } finally {

               lock.unlock(); //釋放鎖

           }

       } else {

           System.out.print("獲取鎖失敗");

       }

 

    }

3、tryLock(long time, TimeUnit unit)
特色:能夠設置獲取鎖的等待時間,如tryLock(4, TimeUnit.SECONDS)等待4秒;能夠在等待過程當中相應中斷;
示例代碼略,參考tryLock()代碼,要注意異常的處理。
4、lockInterruptibly()
特色:當一個使用該方法獲取鎖的線程沒有獲取到鎖處於阻塞等待狀態時,能夠調用線程的interrupt()方法中斷等待。
示例代碼略。

Lock接口有一個實現類ReentrantLock,便可重入鎖,除實現Lock接口的方法外,還提供了很是豐富的其餘的鎖操做方法,如公平鎖和非公平鎖。

·

4、讀寫鎖ReadWriteLock

ReadWriteLock是一個接口,接口代碼:

·

public interface ReadWriteLock {

    Lock readLock(); //讀鎖

 

    Lock writeLock(); //寫鎖

}

裏面只定義了兩個方法,一個獲取讀鎖,一個得到寫鎖,將資源的鎖分開爲兩個,從而使得資源能夠在多線程情境下併發讀操做。

ReentrantReadWriteLock類實現了ReadWriteLock接口,並提供了更多方法,最主要的仍是獲取讀寫鎖的方法。

讀寫鎖代碼示例:

public class LockTest {

    private ReadWriteLock lock = new ReentrantReadWriteLock(); //建立鎖

 

    public static void main(String[] args) {

        final LockTest lockTest = new LockTest();

        new Thread("A") {

            public void run() {

                lockTest.getLock(Thread.currentThread());

            }

        }.start();

        new Thread("B") {

            public void run() {

                lockTest.getLock(Thread.currentThread());

            }

        }.start();

    }

 

    public void getLock(Thread thread) {

 

        lock.readLock().lock(); //得到讀鎖

        try {

            System.out.println("線程" + thread.getName() + "得到讀鎖");

            long startTime = System.currentTimeMillis();

            while (System.currentTimeMillis() - startTime <= 1) {

                System.out.println("線程" + thread.getName() + "進行讀操做......");

            }

 

        } catch (Exception e) {

 

        } finally {

            lock.readLock().unlock();

            System.out.println("線程" + thread.getName() + "釋放讀鎖");

        }

 

    }

}

結果:

5、幾種鎖的概念介紹

1、可重入鎖
具有可重入性的鎖,即爲可重入鎖,好比synchronized和ReentrantLock都是。鎖基於線程分配,當某個線程得到了鎖去執行某個方法,方法中若是再次須要獲取鎖資源時,當前線程能夠直接得到鎖,沒必要從新申請。
2、可中斷鎖
能夠響應中斷的鎖。synchronized不是可中斷鎖,可是Lock是可中斷鎖。
3、公平鎖
儘可能按照請求鎖的順序得到鎖。synchronized是非公平鎖,ReentrantLock和ReentrantReadWriteLock提供了設置公平鎖的方法,不過默認爲非公平鎖。非公平鎖可能致使某些線程永遠獲取不到鎖。
4、讀寫鎖 將對資源的鎖分爲兩個,一個讀鎖和一個寫鎖,使得多個線程之間的讀操做不會發生衝突能夠並行,這極大的提升了讀的效率。不過讀鎖和寫鎖、寫鎖和寫鎖之間都是互斥的。

相關文章
相關標籤/搜索