欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Java Slipped Conditions

 更新時間:2021年01月22日 09:27:05   作者:無才的小易  
這篇文章主要介紹了Java Slipped Conditions的相關(guān)資料,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下

所謂Slipped conditions,就是說, 從一個線程檢查某一特定條件到該線程操作此條件期間,這個條件已經(jīng)被其它線程改變,導(dǎo)致第一個線程在該條件上執(zhí)行了錯誤的操作。這里有一個簡單的例子:

public class Lock {
  private boolean isLocked = true;
  public void lock(){
   synchronized(this){
    while(isLocked){
     try{
      this.wait();
     } catch(InterruptedException e){
      //do nothing, keep waiting
     }
    }
   }
   synchronized(this){
    isLocked = true;
   }
  }
  public synchronized void unlock(){
   isLocked = false;
   this.notify();
  }
}

我們可以看到,lock()方法包含了兩個同步塊。第一個同步塊執(zhí)行wait操作直到isLocked變?yōu)閒alse才退出,第二個同步塊將isLocked置為true,以此來鎖住這個Lock實例避免其它線程通過lock()方法。

我們可以設(shè)想一下,假如在某個時刻isLocked為false, 這個時候,有兩個線程同時訪問lock方法。如果第一個線程先進入第一個同步塊,這個時候它會發(fā)現(xiàn)isLocked為false,若此時允許第二個線程執(zhí)行,它也進入第一個同步塊,同樣發(fā)現(xiàn)isLocked是false?,F(xiàn)在兩個線程都檢查了這個條件為false,然后它們都會繼續(xù)進入第二個同步塊中并設(shè)置isLocked為true。

這個場景就是slipped conditions的例子,兩個線程檢查同一個條件, 然后退出同步塊,因此在這兩個線程改變條件之前,就允許其它線程來檢查這個條件。換句話說,條件被某個線程檢查到該條件被此線程改變期間,這個條件已經(jīng)被其它線程改變過了。

為避免slipped conditions,條件的檢查與設(shè)置必須是原子的,也就是說,在第一個線程檢查和設(shè)置條件期間,不會有其它線程檢查這個條件。

解決上面問題的方法很簡單,只是簡單的把isLocked = true這行代碼移到第一個同步塊中,放在while循環(huán)后面即可:

public class Lock {
  private boolean isLocked = true;
  public void lock(){
   synchronized(this){
    while(isLocked){
     try{
      this.wait();
     } catch(InterruptedException e){
      //do nothing, keep waiting
     }
    }
    isLocked = true;
   }
  }
  public synchronized void unlock(){
   isLocked = false;
   this.notify();
  }
}

現(xiàn)在檢查和設(shè)置isLocked條件是在同一個同步塊中原子地執(zhí)行了。

一個更現(xiàn)實的例子

也許你會說,我才不可能寫這么挫的代碼,還覺得slipped conditions是個相當(dāng)理論的問題。但是第一個簡單的例子只是用來更好的展示slipped conditions。

饑餓和公平中實現(xiàn)的公平鎖也許是個更現(xiàn)實的例子。再看下嵌套管程鎖死中那個幼稚的實現(xiàn),如果我們試圖解決其中的嵌套管程鎖死問題,很容易產(chǎn)生slipped conditions問題。首先讓我們看下嵌套管程鎖死中的例子:

//Fair Lock implementation with nested monitor lockout problem
public class FairLock {
 private boolean isLocked = false;
 private Thread lockingThread = null;
 private List waitingThreads =
      new ArrayList();
 public void lock() throws InterruptedException{
  QueueObject queueObject = new QueueObject();
  synchronized(this){
   waitingThreads.add(queueObject);
   while(isLocked || waitingThreads.get(0) != queueObject){
    synchronized(queueObject){
     try{
      queueObject.wait();
     }catch(InterruptedException e){
      waitingThreads.remove(queueObject);
      throw e;
     }
    }
   }
   waitingThreads.remove(queueObject);
   isLocked = true;
   lockingThread = Thread.currentThread();
  }
 }
 public synchronized void unlock(){
  if(this.lockingThread != Thread.currentThread()){
   throw new IllegalMonitorStateException(
    "Calling thread has not locked this lock");
  }
  isLocked   = false;
  lockingThread = null;
  if(waitingThreads.size() > 0){
   QueueObject queueObject = waitingThread.get(0);
   synchronized(queueObject){
    queueObject.notify();
   }
  }
 }
}1public class QueueObject {}

我們可以看到synchronized(queueObject)及其中的queueObject.wait()調(diào)用是嵌在synchronized(this)塊里面的,這會導(dǎo)致嵌套管程鎖死問題。為避免這個問題,我們必須將synchronized(queueObject)塊移出synchronized(this)塊。移出來之后的代碼可能是這樣的:

//Fair Lock implementation with slipped conditions problem
public class FairLock {
 private boolean isLocked = false;
 private Thread lockingThread = null;
 private List waitingThreads =
      new ArrayList();
 public void lock() throws InterruptedException{
  QueueObject queueObject = new QueueObject();
  synchronized(this){
   waitingThreads.add(queueObject);
  }
  boolean mustWait = true;
  while(mustWait){
   synchronized(this){
    mustWait = isLocked || waitingThreads.get(0) != queueObject;
   }
   synchronized(queueObject){
    if(mustWait){
     try{
      queueObject.wait();
     }catch(InterruptedException e){
      waitingThreads.remove(queueObject);
      throw e;
     }
    }
   }
  }
  synchronized(this){
   waitingThreads.remove(queueObject);
   isLocked = true;
   lockingThread = Thread.currentThread();
  }
 }
}

注意:因為我只改動了lock()方法,這里只展現(xiàn)了lock方法。

現(xiàn)在lock()方法包含了3個同步塊。

第一個,synchronized(this)塊通過mustWait = isLocked || waitingThreads.get(0) != queueObject檢查內(nèi)部變量的值。

第二個,synchronized(queueObject)塊檢查線程是否需要等待。也有可能其它線程在這個時候已經(jīng)解鎖了,但我們暫時不考慮這個問題。我們就假設(shè)這個鎖處在解鎖狀態(tài),所以線程會立馬退出synchronized(queueObject)塊。

第三個,synchronized(this)塊只會在mustWait為false的時候執(zhí)行。它將isLocked重新設(shè)回true,然后離開lock()方法。

設(shè)想一下,在鎖處于解鎖狀態(tài)時,如果有兩個線程同時調(diào)用lock()方法會發(fā)生什么。首先,線程1會檢查到isLocked為false,然后線程2同樣檢查到isLocked為false。接著,它們都不會等待,都會去設(shè)置isLocked為true。這就是slipped conditions的一個最好的例子。

解決Slipped Conditions問題

要解決上面例子中的slipped conditions問題,最后一個synchronized(this)塊中的代碼必須向上移到第一個同步塊中。為適應(yīng)這種變動,代碼需要做點小改動。下面是改動過的代碼:

//Fair Lock implementation without nested monitor lockout problem,
//but with missed signals problem.
public class FairLock {
 private boolean isLocked = false;
 private Thread lockingThread = null;
 private List waitingThreads =
      new ArrayList();
 public void lock() throws InterruptedException{
  QueueObject queueObject = new QueueObject();
  synchronized(this){
   waitingThreads.add(queueObject);
  }
  boolean mustWait = true;
  while(mustWait){
   synchronized(this){
    mustWait = isLocked || waitingThreads.get(0) != queueObject;
    if(!mustWait){
     waitingThreads.remove(queueObject);
     isLocked = true;
     lockingThread = Thread.currentThread();
     return;
    }
   }  
   synchronized(queueObject){
    if(mustWait){
     try{
      queueObject.wait();
     }catch(InterruptedException e){
      waitingThreads.remove(queueObject);
      throw e;
     }
    }
   }
  }
 }
}

我們可以看到對局部變量mustWait的檢查與賦值是在同一個同步塊中完成的。還可以看到,即使在synchronized(this)塊外面檢查了mustWait,在while(mustWait)子句中,mustWait變量從來沒有在synchronized(this)同步塊外被賦值。當(dāng)一個線程檢查到mustWait是false的時候,它將自動設(shè)置內(nèi)部的條件(isLocked),所以其它線程再來檢查這個條件的時候,它們就會發(fā)現(xiàn)這個條件的值現(xiàn)在為true了。

synchronized(this)塊中的return;語句不是必須的。這只是個小小的優(yōu)化。如果一個線程肯定不會等待(即mustWait為false),那么就沒必要讓它進入到synchronized(queueObject)同步塊中和執(zhí)行if(mustWait)子句了。

細心的讀者可能會注意到上面的公平鎖實現(xiàn)仍然有可能丟失信號。設(shè)想一下,當(dāng)該FairLock實例處于鎖定狀態(tài)時,有個線程來調(diào)用lock()方法。執(zhí)行完第一個 synchronized(this)塊后,mustWait變量的值為true。再設(shè)想一下調(diào)用lock()的線程是通過搶占式的,擁有鎖的那個線程那個線程此時調(diào)用了unlock()方法,但是看下之前的unlock()的實現(xiàn)你會發(fā)現(xiàn),它調(diào)用了queueObject.notify()。但是,因為lock()中的線程還沒有來得及調(diào)用queueObject.wait(),所以queueObject.notify()調(diào)用也就沒有作用了,信號就丟失掉了。如果調(diào)用lock()的線程在另一個線程調(diào)用queueObject.notify()之后調(diào)用queueObject.wait(),這個線程會一直阻塞到其它線程調(diào)用unlock方法為止,但這永遠也不會發(fā)生。

公平鎖實現(xiàn)的信號丟失問題在饑餓和公平一文中我們已有過討論,把QueueObject轉(zhuǎn)變成一個信號量,并提供兩個方法:doWait()和doNotify()。這些方法會在QueueObject內(nèi)部對信號進行存儲和響應(yīng)。用這種方式,即使doNotify()在doWait()之前調(diào)用,信號也不會丟失。

以上就是詳解Java Slipped Conditions的詳細內(nèi)容,更多關(guān)于Java Slipped Conditions的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 詳解Huffman編碼算法之Java實現(xiàn)

    詳解Huffman編碼算法之Java實現(xiàn)

    Huffman編碼是一種編碼方式,常用于無損壓縮。本文只介紹用Java語言來實現(xiàn)該編碼方式的算法和數(shù)據(jù)結(jié)構(gòu)。有興趣的可以了解一下。
    2016-12-12
  • spring boot中使用@Async實現(xiàn)異步調(diào)用任務(wù)

    spring boot中使用@Async實現(xiàn)異步調(diào)用任務(wù)

    本篇文章主要介紹了spring boot中使用@Async實現(xiàn)異步調(diào)用任務(wù),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-02-02
  • java入門概念個人理解之package與import淺析

    java入門概念個人理解之package與import淺析

    下面小編就為大家?guī)硪黄猨ava入門概念個人理解之package與import淺析。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-08-08
  • 淺談Redis持久化的幾種方式

    淺談Redis持久化的幾種方式

    這篇文章主要介紹了淺談Redis持久化的幾種方式,前面說到了Redis持久化的 實現(xiàn)方式主要分為了:快照持久化(RDB)、寫日志持久化(AOF)
    ,其中快照持久化方式也就是RDB ,需要的朋友可以參考下
    2023-08-08
  • 詳解SpringMVC攔截器(資源和權(quán)限管理)

    詳解SpringMVC攔截器(資源和權(quán)限管理)

    本篇文章主要介紹了SpringMVC攔截器(資源和權(quán)限管理),具有一定的參考價值,有興趣的可以了解一下。
    2016-12-12
  • 詳解SpringBoot如何創(chuàng)建自定義Starter

    詳解SpringBoot如何創(chuàng)建自定義Starter

    Spring Boot的自動配置機制為開發(fā)人員提供了一種輕松集成和配置各種功能的便捷方式,本文將深入探討在Spring Boot中如何創(chuàng)建自定義Starter,為構(gòu)建模塊化且易維護的應(yīng)用提供有力的支持,需要的朋友可以參考下
    2024-02-02
  • jmeter+ant+jenkins自動化測試環(huán)境配置搭建過程

    jmeter+ant+jenkins自動化測試環(huán)境配置搭建過程

    在搭建jmeter+ant+jenkins環(huán)境有些前提條件,那就是要先配置好java環(huán)境、安裝好jenkins以及配置好jmeter,這樣才能省去很多的事情,對jmeter+ant+jenkins自動化測試環(huán)境配置搭建過程感興趣的朋友一起看看吧
    2021-12-12
  • Java如何將ResultSet結(jié)果集遍歷到List中

    Java如何將ResultSet結(jié)果集遍歷到List中

    這篇文章主要介紹了Java如何將ResultSet結(jié)果集遍歷到List中問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • Java GZip 基于磁盤實現(xiàn)壓縮和解壓的方法

    Java GZip 基于磁盤實現(xiàn)壓縮和解壓的方法

    這篇文章主要介紹了Java GZip 基于磁盤實現(xiàn)壓縮和解壓,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考需要的朋友可以參考下
    2020-08-08
  • Java基礎(chǔ)教程_判斷語句if else

    Java基礎(chǔ)教程_判斷語句if else

    下面小編就為大家?guī)硪黄狫ava基礎(chǔ)教程_判斷語句if else。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-06-06

最新評論