国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Java中的鎖

paulquei / 2985人閱讀

摘要:中的鎖鎖像同步塊一樣,是一種線程同步機制,但比中的同步塊更復雜。這意味著如果一個線程進入了代碼中的同步塊,并因此獲得了該同步塊使用的同步對象對應的管程上的鎖,那么這個線程可以進入由同一個管程對象所同步的另一個代碼塊。

Java中的鎖

鎖像synchronized同步塊一樣,是一種線程同步機制,但比Java中的synchronized同步塊更復雜。因為鎖(以及其它更高級的線程同步機制)是由synchronized同步塊的方式實現的,所以我們還不能完全擺脫synchronized關鍵字(譯者注:這說的是Java 5之前的情況)。

自Java 5開始,java.util.concurrent.locks包中包含了一些鎖的實現,因此你不用去實現自己的鎖了。但是你仍然需要去了解怎樣使用這些鎖,且了解這些實現背后的理論也是很有用處的。可以參考我對java.util.concurrent.locks.Lock的介紹,以了解更多關于鎖的信息。

一個簡單的鎖

讓我們從java中的一個同步塊開始:

public class Counter{
    private int count = 0;

    public int inc(){
        synchronized(this){
            return ++count;
        }
    }
}

可以看到在inc()方法中有一個synchronized(this)代碼塊。該代碼塊可以保證在同一時間只有一個線程可以執行return ++count。雖然在synchronized的同步塊中的代碼可以更加復雜,但是++count這種簡單的操作已經足以表達出線程同步的意思。

以下的Counter類用Lock代替synchronized達到了同樣的目的:

public class Counter{
    private Lock lock = new Lock();
    private int count = 0;

    public int inc(){
        lock.lock();
        int newCount = ++count;
        lock.unlock();
        return newCount;
    }
}

lock()方法會對Lock實例對象進行加鎖,因此所有對該對象調用lock()方法的線程都會被阻塞,直到該Lock對象的unlock()方法被調用。

這里有一個Lock類的簡單實現:

public class Counter{
public class Lock{
    private boolean isLocked = false;

    public synchronized void lock()
        throws InterruptedException{
        while(isLocked){
            wait();
        }
        isLocked = true;
    }

    public synchronized void unlock(){
        isLocked = false;
        notify();
    }
}

注意其中的while(isLocked)循環,它又被叫做“自旋鎖”。自旋鎖以及wait()和notify()方法在線程通信這篇文章中有更加詳細的介紹。當isLocked為true時,調用lock()的線程在wait()調用上阻塞等待。為防止該線程沒有收到notify()調用也從wait()中返回(也稱作虛假喚醒),這個線程會重新去檢查isLocked條件以決定當前是否可以安全地繼續執行還是需要重新保持等待,而不是認為線程被喚醒了就可以安全地繼續執行了。如果isLocked為false,當前線程會退出while(isLocked)循環,并將isLocked設回true,讓其它正在調用lock()方法的線程能夠在Lock實例上加鎖。

當線程完成了臨界區(位于lock()和unlock()之間)中的代碼,就會調用unlock()。執行unlock()會重新將isLocked設置為false,并且通知(喚醒)其中一個(若有的話)在lock()方法中調用了wait()函數而處于等待狀態的線程。

鎖的可重入性

Java中的synchronized同步塊是可重入的。這意味著如果一個java線程進入了代碼中的synchronized同步塊,并因此獲得了該同步塊使用的同步對象對應的管程上的鎖,那么這個線程可以進入由同一個管程對象所同步的另一個java代碼塊。下面是一個例子:

public class Reentrant{
    public synchronized outer(){
        inner();
    }

    public synchronized inner(){
        //do something
    }
}

注意outer()和inner()都被聲明為synchronized,這在Java中和synchronized(this)塊等效。如果一個線程調用了outer(),在outer()里調用inner()就沒有什么問題,因為這兩個方法(代碼塊)都由同一個管程對象(”this”)所同步。如果一個線程已經擁有了一個管程對象上的鎖,那么它就有權訪問被這個管程對象同步的所有代碼塊。這就是可重入。線程可以進入任何一個它已經擁有的鎖所同步著的代碼塊。

前面給出的鎖實現不是可重入的。如果我們像下面這樣重寫Reentrant類,當線程調用outer()時,會在inner()方法的lock.lock()處阻塞住。

public class Reentrant2{
    Lock lock = new Lock();

    public outer(){
        lock.lock();
        inner();
        lock.unlock();
    }

    public synchronized inner(){
        lock.lock();
        //do something
        lock.unlock();
    }
}

調用outer()的線程首先會鎖住Lock實例,然后繼續調用inner()。inner()方法中該線程將再一次嘗試鎖住Lock實例,結果該動作會失敗(也就是說該線程會被阻塞),因為這個Lock實例已經在outer()方法中被鎖住了。

兩次lock()之間沒有調用unlock(),第二次調用lock就會阻塞,看過lock()實現后,會發現原因很明顯:

public class Lock{
    boolean isLocked = false;

    public synchronized void lock()
        throws InterruptedException{
        while(isLocked){
            wait();
        }
        isLocked = true;
    }

    ...
}

一個線程是否被允許退出lock()方法是由while循環(自旋鎖)中的條件決定的。當前的判斷條件是只有當isLocked為false時lock操作才被允許,而沒有考慮是哪個線程鎖住了它。

為了讓這個Lock類具有可重入性,我們需要對它做一點小的改動:

public class Lock{
    boolean isLocked = false;
    Thread  lockedBy = null;
    int lockedCount = 0;

    public synchronized void lock()
        throws InterruptedException{
        Thread callingThread =
            Thread.currentThread();
        while(isLocked && lockedBy != callingThread){
            wait();
        }
        isLocked = true;
        lockedCount++;
        lockedBy = callingThread;
  }

    public synchronized void unlock(){
        if(Thread.curentThread() ==
            this.lockedBy){
            lockedCount--;

            if(lockedCount == 0){
                isLocked = false;
                notify();
            }
        }
    }

    ...
}

注意到現在的while循環(自旋鎖)也考慮到了已鎖住該Lock實例的線程。如果當前的鎖對象沒有被加鎖(isLocked = false),或者當前調用線程已經對該Lock實例加了鎖,那么while循環就不會被執行,調用lock()的線程就可以退出該方法(譯者注:“被允許退出該方法”在當前語義下就是指不會調用wait()而導致阻塞)

除此之外,我們需要記錄同一個線程重復對一個鎖對象加鎖的次數。否則,一次unblock()調用就會解除整個鎖,即使當前鎖已經被加鎖過多次。在unlock()調用沒有達到對應lock()調用的次數之前,我們不希望鎖被解除。

現在這個Lock類就是可重入的了。

鎖的公平性

Java的synchronized塊并不保證嘗試進入它們的線程的順序。因此,如果多個線程不斷競爭訪問相同的synchronized同步塊,就存在一種風險,其中一個或多個線程永遠也得不到訪問權——也就是說訪問權總是分配給了其它線程。這種情況被稱作線程饑餓。為了避免這種問題,鎖需要實現公平性。本文所展現的鎖在內部是用synchronized同步塊實現的,因此它們也不保證公平性。饑餓和公平中有更多關于該內容的討論。

在finally語句中調用unlock()

如果用Lock來保護臨界區,并且臨界區有可能會拋出異常,那么在finally語句中調用unlock()就顯得非常重要了。這樣可以保證這個鎖對象可以被解鎖以便其它線程能繼續對其加鎖。以下是一個示例:

lock.lock();
try{
    //do critical section code,
    //which may throw exception
} finally {
    lock.unlock();
}

這個簡單的結構可以保證當臨界區拋出異常時Lock對象可以被解鎖。如果不是在finally語句中調用的unlock(),當臨界區拋出異常時,Lock對象將永遠停留在被鎖住的狀態,這會導致其它所有在該Lock對象上調用lock()的線程一直阻塞。

原文 locks
譯者 申章 校對 丁一
via ifeve

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/69764.html

相關文章

  • java并發編程學習1--基礎知識

    摘要:死亡狀態線程退出有可能是正常執行完成也有可能遇見異常退出。類有新建與死亡狀態返回其余狀態返回判斷線程是否存活。線程因某些原因進入阻塞狀態。執行同步代碼塊的過程中執行了當前線程放棄開始睡眠進入就緒狀態但是不會釋放鎖。 【java內存模型簡介 JVM中存在一個主存區(Main Memory或Java Heap Memory),Java中所有變量都是存在主存中的,對于所有線程進行共享,而每個...

    huangjinnan 評論0 收藏0
  • Java的鎖之樂觀鎖與悲觀鎖

    摘要:解決上問題在變量前添加版本號,將變成循環時間長開銷大,因為自旋需要消耗只能保證一個共享變量的原子操作分類二重入鎖支持重進入的鎖,排它鎖分類三讀寫鎖一對鎖,讀鎖,寫鎖,在同一時刻允許多線程訪問 1、 分類一:樂觀鎖與悲觀鎖   a)悲觀鎖:認為其他線程會干擾本身線程操作,所以加鎖 i.具體表現形式:synchronized關鍵字和lock實現類 ...

    huangjinnan 評論0 收藏0
  • Java的鎖

    摘要:當前線程在超時時間內被中斷超時時間結束,返回釋放鎖獲取等待通知組件,該組件和當前的鎖綁定,當前線程只有獲取了鎖,才能調用該組件的方法,調用后,當前線程將釋放鎖。同步器是實現鎖的關鍵,在鎖的實現中聚合同步器,利用同步器實現鎖的語義。 本文在參考java并發編程實戰后完成,參考內容較多 Java中的鎖 鎖是用來控制多線程訪問共享資源的方式,一個鎖能夠防止多個線程同事訪問共享資源。在Lock...

    gaara 評論0 收藏0
  • 淺談Java并發編程系列(七) —— 深入解析synchronized關鍵字

    摘要:第一個字被稱為。經量級鎖的加鎖過程當一個對象被鎖定時,被復制到當前嘗試獲取鎖的線程的線程棧的鎖記錄空間被復制的官方稱為。根據鎖對象目前是否處于被鎖定狀態,撤銷偏向后恢復到未鎖定或經量級鎖定狀態。 Synchronized關鍵字 synchronized的鎖機制的主要優勢是Java語言內置的鎖機制,因此,JVM可以自由的優化而不影響已存在的代碼。 任何對象都擁有對象頭這一數據結構來支持鎖...

    piglei 評論0 收藏0
  • Java并發總結

    摘要:限期阻塞調用方法等待時間結束或線程執行完畢。終止狀態線程執行完畢或出現異常退了。和都會檢查線程何時中斷,并且在發現中斷時提前放回。工廠方法將線程池的最大大小設置為,而將基本大小設置為,并將超時大小設置為分鐘。 wait()、notify()、notifyAll() Object是所有類的基類,它有5個方法組成了等待、通知機制的核心:notify()、notifyAll()、wait()...

    szysky 評論0 收藏0

發表評論

0條評論

paulquei

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<