亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? 多線程java程序中常見錯誤的巧處理.txt

?? 含有許多JAVA的技巧!
?? TXT
字號:
多線程Java程序中常見錯誤的巧處理 
(加入日期:2003-4-13 點擊數:2261)
【對此文發表評論】 【編程愛好者論壇】 【保存文章至硬盤】 【打印文章】 
 
作者:俞良松 

  在幾乎所有編程語言中,由于多線程引發的錯誤都有著難以再現的特點,程序的死鎖或其它多線程錯誤可能只在某些特殊的情形下才出現,或在不同的VM上運行同一個程序時錯誤表現不同。因此,在編寫多線程程序時,事先認識和防范可能出現的錯誤特別重要。 

  無論是客戶端還是服務器端多線程Java程序,最常見的多線程問題包括死鎖、隱性死鎖和數據競爭。 

  死鎖 

  死鎖是這樣一種情形:多個線程同時被阻塞,它們中的一個或者全部都在等待某個資源被釋放。由于線程被無限期地阻塞,因此程序不可能正常終止。 

  導致死鎖的根源在于不適當地運用“synchronized”關鍵詞來管理線程對特定對象的訪問?!皊ynchronized”關鍵詞的作用是,確保在某個時刻只有一個線程被允許執行特定的代碼塊,因此,被允許執行的線程首先必須擁有對變量或對象的排他性的訪問權。當線程訪問對象時,線程會給對象加鎖,而這個鎖導致其它也想訪問同一對象的線程被阻塞,直至第一個線程釋放它加在對象上的鎖。 

  由于這個原因,在使用“synchronized”關鍵詞時,很容易出現兩個線程互相等待對方做出某個動作的情形。代碼一是一個導致死鎖的簡單例子。 

//代碼一
class Deadlocker {
int field_1;
private Object lock_1 = new int[1];
int field_2;
private Object lock_2 = new int[1];

public void method1(int value) {
  “synchronized” (lock_1) {
   “synchronized” (lock_2) {
    field_1 = 0; field_2 = 0;
   }
  }
}

public void method2(int value) {
  “synchronized” (lock_2) {
   “synchronized” (lock_1) {
    field_1 = 0; field_2 = 0;
   }
  }
}
}



  參考代碼一,考慮下面的過程: 

  ◆ 一個線程(ThreadA)調用method1()。 

  ◆ ThreadA在lock_1上同步,但允許被搶先執行。 

  ◆ 另一個線程(ThreadB)開始執行。 

  ◆ ThreadB調用method2()。 

  ◆ ThreadB獲得lock_2,繼續執行,企圖獲得lock_1。但ThreadB不能獲得lock_1,因為ThreadA占有lock_1。 

  ◆ 現在,ThreadB阻塞,因為它在等待ThreadA釋放lock_1。 

  ◆ 現在輪到ThreadA繼續執行。ThreadA試圖獲得lock_2,但不能成功,因為lock_2已經被ThreadB占有了。 

  ◆ ThreadA和ThreadB都被阻塞,程序死鎖。 

  當然,大多數的死鎖不會這么顯而易見,需要仔細分析代碼才能看出,對于規模較大的多線程程序來說尤其如此。好的線程分析工具,例如JProbe Threadalyzer能夠分析死鎖并指出產生問題的代碼位置。 

  隱性死鎖 

  隱性死鎖由于不規范的編程方式引起,但不一定每次測試運行時都會出現程序死鎖的情形。由于這個原因,一些隱性死鎖可能要到應用正式發布之后才會被發現,因此它的危害性比普通死鎖更大。下面介紹兩種導致隱性死鎖的情況:加鎖次序和占有并等待。 

  加鎖次序 

  當多個并發的線程分別試圖同時占有兩個鎖時,會出現加鎖次序沖突的情形。如果一個線程占有了另一個線程必需的鎖,就有可能出現死鎖??紤]下面的情形,ThreadA和ThreadB兩個線程分別需要同時擁有lock_1、lock_2兩個鎖,加鎖過程可能如下: 

  ◆ ThreadA獲得lock_1; 

  ◆ ThreadA被搶占,VM調度程序轉到ThreadB; 

  ◆ ThreadB獲得lock_2; 

  ◆ ThreadB被搶占,VM調度程序轉到ThreadA; 

  ◆ ThreadA試圖獲得lock_2,但lock_2被ThreadB占有,所以ThreadA阻塞; 

  ◆ 調度程序轉到ThreadB; 

  ◆ ThreadB試圖獲得lock_1,但lock_1被ThreadA占有,所以ThreadB阻塞; 

  ◆ ThreadA和ThreadB死鎖。 

  必須指出的是,在代碼絲毫不做變動的情況下,有些時候上述死鎖過程不會出現,VM調度程序可能讓其中一個線程同時獲得lock_1和lock_2兩個鎖,即線程獲取兩個鎖的過程沒有被中斷。在這種情形下,常規的死鎖檢測很難確定錯誤所在。 

  占有并等待 

  如果一個線程獲得了一個鎖之后還要等待來自另一個線程的通知,可能出現另一種隱性死鎖,考慮代碼二。 

//代碼二
public class queue {
static java.lang.Object queueLock_;
Producer producer_;
Consumer consumer_;

public class Producer {
  void produce() {
   while (!done) {
    “synchronized” (queueLock_) {
     produceItemAndAddItToQueue();
     “synchronized” (consumer_) {
      consumer_.notify();
     }
    }
   }
  }

  public class Consumer {
   consume() {
    while (!done) {
     “synchronized” (queueLock_) {
      “synchronized” (consumer_) {
       consumer_.wait();
      }
      removeItemFromQueueAndProcessIt();
     }
    }
   }
  }
}
}



  在代碼二中,Producer向隊列加入一項新的內容后通知Consumer,以便它處理新的內容。問題在于,Consumer可能保持加在隊列上的鎖,阻止Producer訪問隊列,甚至在Consumer等待Producer的通知時也會繼續保持鎖。這樣,由于Producer不能向隊列添加新的內容,而Consumer卻在等待Producer加入新內容的通知,結果就導致了死鎖。 

  在等待時占有的鎖是一種隱性的死鎖,這是因為事情可能按照比較理想的情況發展—Producer線程不需要被Consumer占據的鎖。盡管如此,除非有絕對可靠的理由肯定Producer線程永遠不需要該鎖,否則這種編程方式仍是不安全的。有時“占有并等待”還可能引發一連串的線程等待,例如,線程A占有線程B需要的鎖并等待,而線程B又占有線程C需要的鎖并等待等。 

  要改正代碼二的錯誤,只需修改Consumer類,把wait()移出“synchronized”()即可。 

  數據競爭 

  數據競爭是由于訪問共享資源(例如變量)時缺乏或不適當地運用同步機制引起。如果沒有正確地限定某一時刻某一個線程可以訪問變量,就會出現數據競爭,此時贏得競爭的線程獲得訪問許可,但會導致不可預知的結果。 

  由于線程的運行可以在任何時候被中斷(即運行機會被其它線程搶占),所以不能假定先開始運行的線程總是比后開始運行的線程先訪問到兩者共享的數據。另外,在不同的VM上,線程的調度方式也可能不同,從而使數據競爭問題更加復雜。 

  有時,數據競爭不會影響程序的最終運行結果,但在另一些時候,有可能導致不可預料的結果。 

  良性數據競爭 

  并非所有的數據競爭都是錯誤??紤]代碼三的例子。假設getHouse()向所有的線程返回同一House,可以看出,這里會出現競爭:BrickLayer從House.foundationReady_讀取,而FoundationPourer寫入到House.foundationReady_。 

//代碼三
public class House {
public volatile boolean foundationReady_ = false;
}

public class FoundationPourer extends Thread {
public void run() {
  House a = getHouse();
  a.foundationReady_ = true;
}
}

public class BrickLayer extends Thread {
public void run() {
  House a = getHouse();
   while (!a.foundationReady_) {
    try {
     Thread.sleep(500);
    }
    catch (Exception e) {
     System.err.println(“Exception:” + e);
    }
   }
  }
} 
}



  盡管存在競爭,但根據Java VM規范,Boolean數據的讀取和寫入都是原則性的,也就是說,VM不能中斷線程的讀取或寫入操作。一旦數據改動成功,不存在將它改回原來數據的必要(不需要“回退”),所以代碼三的數據競爭是良性競爭,代碼是安全的。 

  惡性數據競爭 

  首先看一下代碼四的例子。 

//代碼四
public class Account {
private int balance_; // 賬戶余額
public int getBalance(void) {
  return balance_;
}
public void setBalance(int setting) {
  balance_ = setting;
}
}

public class CustomerInfo {
private int numAccounts_;
private Account[] accounts_;
public void withdraw(int accountNumber, int amount) {
  int temp = accounts_[accountNumber].getBalance();
  temp = temp - amount;
  accounts_[accountNumber].setBalance(temp);
}
public void deposit(int accountNumber, int amount) {
  int temp = accounts_[accountNumber].getBalance();
  temp = temp + amount;
  accounts_[accountNumber].setBalance(temp);
}
}



  如果丈夫A和妻子B試圖通過不同的銀行柜員機同時向同一賬戶存錢,會發生什么事情?讓我們假設賬戶的初始余額是100元,看看程序的一種可能的執行經過。 

  B存錢25元,她的柜員機開始執行deposit()。首先取得當前余額100,把這個余額保存在本地的臨時變量,然后把臨時變量加25,臨時變量的值變成125。現在,在調用setBalance()之前,線程調度器中斷了該線程。 

  A存入50元。當B的線程仍處于掛起狀態時,A這面開始執行deposit():getBalance()返回100(因為這時B的線程尚未把修改后的余額寫入),A的線程在現有余額的基礎上加50得到150,并把150這個值保存到臨時變量。接著,A的線程在調用setBalance()之前,也被中斷執行。 

  現在,B的線程接著運行,把保存在臨時變量中的值(125)寫入到余額,柜員機告訴B說交易完成,賬戶余額是125元。接下來,A的線程繼續運行,把臨時變量的值(150)寫入到余額,柜員機告訴A說交易完成,賬戶余額是150元。 

  最后得到的結果是什么?B的存款消失不見,就像B根本沒有存過錢一樣。 

  也許有人會認為,可以把getBalance()和setBalance()改成同步方法保護Account.balance_,解決數據競爭問題。其實這種辦法是行不通的?!皊ynchronized”關鍵詞可以確保同一時刻只有一個線程執行getBalance()或setBalance()方法,但這不能在一個線程操作期間阻止另一個線程修改賬戶余額。 

要正確運用“synchronized”關鍵詞,就必須認識到這里要保護的是整個交易過程不被另一個線程干擾,而不僅僅是對數據訪問的某一個步驟進行保護。 

  所以,本例的關鍵是當一個線程獲得當前余額之后,要保證其它的線程不能修改余額,直到第一個線程的余額處理工作全部完成。正確的修改方法是把deposit()和withdraw()改成同步方法。 

  死鎖、隱性死鎖和數據競爭是Java多線程編程中最常見的錯誤。要寫出健壯的多線程代碼,正確理解和運用“synchronized”關鍵詞是很重要的。另外,好的線程分析工具,例如JProbe Threadalyzer,能夠極大地簡化錯誤檢測。對于分析那些不一定每次執行時都會出現的錯誤,分析工具尤其有用。 

本欄文章均來自于互聯網,版權歸原作者和各發布網站所有,本站收集這些文章僅供學習參考之用。任何人都不能將這些文章用于商業或者其他目的。( ProgramFan.Com ) 

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
日韩av一区二区三区四区| 国产在线日韩欧美| 美女视频黄频大全不卡视频在线播放| 激情文学综合网| 色哟哟一区二区三区| 精品国产91亚洲一区二区三区婷婷| 国产精品美日韩| 蜜桃精品在线观看| 在线观看三级视频欧美| 国产欧美一区二区三区沐欲| 日韩精品1区2区3区| 91蜜桃网址入口| 久久精品视频一区二区三区| 青青草91视频| 欧美日韩一区久久| 亚洲精品欧美激情| 粉嫩aⅴ一区二区三区四区| 日韩一级免费观看| 亚洲成人综合视频| 在线观看网站黄不卡| 国产精品麻豆久久久| 国产乱色国产精品免费视频| 日韩一区二区三区视频| 亚洲午夜电影网| 91免费在线播放| 国产精品久久网站| 国产成人精品免费网站| 一区二区三区在线高清| 福利视频网站一区二区三区| 精品国产凹凸成av人网站| 日韩av不卡一区二区| 欧美日韩精品一区二区三区四区| 亚洲精品中文字幕乱码三区| 99精品桃花视频在线观看| 国产精品私人影院| av在线一区二区| 亚洲欧美在线视频观看| 暴力调教一区二区三区| 国产精品丝袜一区| 99久久综合国产精品| 日韩一区中文字幕| 一本色道久久综合精品竹菊| 亚洲另类在线一区| 在线视频观看一区| 亚洲第一激情av| 91精品国产91久久久久久最新毛片| 亚洲aⅴ怡春院| 欧美一级高清片| 狠狠色2019综合网| 国产欧美一区二区三区沐欲| www.爱久久.com| 亚洲制服丝袜av| 5858s免费视频成人| 韩国女主播一区二区三区| 国产清纯美女被跳蛋高潮一区二区久久w| 国产一区二区三区不卡在线观看| 久久综合一区二区| 成人免费高清视频在线观看| 亚洲人精品午夜| 欧美性淫爽ww久久久久无| 青青草精品视频| 中文字幕欧美区| 欧美日免费三级在线| 美腿丝袜亚洲三区| 国产精品网站一区| 在线一区二区三区四区五区| 日韩高清中文字幕一区| 久久青草欧美一区二区三区| 99re热这里只有精品免费视频| 亚洲国产一区二区a毛片| 亚洲国产成人午夜在线一区 | zzijzzij亚洲日本少妇熟睡| 亚洲摸摸操操av| 欧美va亚洲va香蕉在线| 99国产精品久久久| 免费观看一级特黄欧美大片| 国产精品女同互慰在线看| 欧美日韩一区中文字幕| 国产剧情一区二区| 亚洲午夜在线观看视频在线| 国产日本欧美一区二区| 欧美浪妇xxxx高跟鞋交| av电影天堂一区二区在线| 欧美aaa在线| 亚洲欧美日韩国产中文在线| 精品人伦一区二区色婷婷| 在线一区二区三区做爰视频网站| 国模娜娜一区二区三区| 亚洲国产精品久久一线不卡| 久久久99精品免费观看不卡| 7777精品伊人久久久大香线蕉完整版 | 日韩国产精品久久久久久亚洲| 久久精品亚洲国产奇米99| 制服丝袜国产精品| 色偷偷成人一区二区三区91 | 污片在线观看一区二区| 国产精品久久久久aaaa樱花| 精品国产sm最大网站免费看| 欧美精选一区二区| 欧美三级资源在线| 91在线丨porny丨国产| 国产91色综合久久免费分享| 精品一区二区精品| 日韩二区三区四区| 亚洲18女电影在线观看| 亚洲精品第一国产综合野| 中文字幕亚洲一区二区va在线| 精品成人一区二区| 精品久久久久av影院| 欧美一级片免费看| 56国语精品自产拍在线观看| 欧美日韩高清在线播放| 欧美视频完全免费看| 欧洲一区在线观看| 色狠狠色噜噜噜综合网| 91福利在线播放| 欧美性色黄大片| 欧美日韩视频不卡| 国产精品嫩草影院av蜜臀| 国产欧美日韩麻豆91| 欧美国产国产综合| 国产女人18毛片水真多成人如厕| 国产喷白浆一区二区三区| 国产网红主播福利一区二区| 国产喂奶挤奶一区二区三区| 国产精品免费丝袜| 中文字幕一区视频| 亚洲美女视频一区| 亚洲综合网站在线观看| 丝袜美腿亚洲一区二区图片| 久久精品理论片| 国产精品18久久久久久久网站| 国产成人午夜高潮毛片| 91啪亚洲精品| 欧美日韩国产经典色站一区二区三区 | 蜜臀av性久久久久蜜臀aⅴ四虎| 免费观看久久久4p| 国产一区二区按摩在线观看| 波多野结衣视频一区| 日本高清不卡aⅴ免费网站| 欧美日韩精品一区二区天天拍小说| 911精品国产一区二区在线| 欧美成人aa大片| 亚洲国产精品t66y| 亚洲18色成人| 国产福利一区二区三区| 91麻豆免费视频| 91精品在线观看入口| 久久亚洲精品国产精品紫薇| 1024精品合集| 日韩电影网1区2区| 成人免费高清视频在线观看| 欧美日韩免费一区二区三区视频| 精品国产区一区| 亚洲日本成人在线观看| 久久99九九99精品| 99国产精品国产精品久久| 91精品国产91综合久久蜜臀| 欧美国产日本韩| 日本在线观看不卡视频| 99热国产精品| 精品日韩欧美在线| 一区二区三区欧美激情| 国产伦精品一区二区三区免费迷| 91国偷自产一区二区三区成为亚洲经典 | 日韩视频免费直播| 国产亚洲一区二区三区在线观看| 一区二区三区日韩欧美| 风间由美一区二区av101| 欧美日韩一级视频| ㊣最新国产の精品bt伙计久久| 捆绑紧缚一区二区三区视频| 91豆麻精品91久久久久久| 久久精品亚洲麻豆av一区二区 | 日韩电影在线免费看| 99精品热视频| 久久久久久久久蜜桃| 青青草原综合久久大伊人精品优势| 色噜噜久久综合| 国产精品拍天天在线| 国产一二精品视频| 日韩手机在线导航| 日韩专区一卡二卡| 91污片在线观看| 国产精品麻豆欧美日韩ww| 国产成人久久精品77777最新版本| 欧美精品aⅴ在线视频| 亚洲午夜激情网站| 色婷婷香蕉在线一区二区| 亚洲欧洲一区二区三区| 国产成人精品网址| 久久久久久久久久电影| 久久国产精品色| 日韩欧美区一区二| 免费成人小视频| 欧美卡1卡2卡| 日本不卡一二三区黄网| 欧美一区二区在线不卡| 亚洲成人精品一区二区| 欧美日韩免费观看一区三区|