java編程多線程并發(fā)處理實例解析
本文主要是通過一個銀行用戶取錢的實例,演示java編程多線程并發(fā)處理場景,具體如下。
從一個例子入手:實現(xiàn)一個銀行賬戶取錢場景的實例代碼。
第一個類:Account.java
賬戶類:
package cn.edu.byr.test;
public class Account {
private String accountNo;
private double balance;
public Account(){
}
public Account(String accountNo,double balance){
this.accountNo = accountNo;
this.balance = balance;
}
public int hashcode(){
return accountNo.hashCode();
}
public String getAccountNo(){
return this.accountNo;
}
public double getBalance(){
return this.balance;
}
public void setBalance(double balance){
this.balance = balance;
}
public Boolean equals(Object obj){
if(this == obj)
return true;
if(obj != null && obj.getClass() == Account.class){
Account target = (Account)obj;
return target.getAccountNo().equals(accountNo);
}
return false;
}
}
第二個類:DrawThread.java
取錢線程類:
package cn.edu.byr.test;
public class DrawThread extends Thread {
private Account account;
private double drawAmount;
public DrawThread(String name,Account account,double drawAmount){
super(name);
this.account = account;
this.drawAmount = drawAmount;
}
public void run(){
// synchronized (account) {
if(account.getBalance() > drawAmount){
System.out.println(getName() + "取錢成功,吐出鈔票:" + drawAmount);
// try{
// Thread.sleep(1);
// }
// catch(InterruptedException e){
// e.printStackTrace();
// }
account.setBalance(account.getBalance() - drawAmount);
System.out.println("\t 余額為 : " + account.getBalance());
} else
System.out.println(getName() + "取錢失敗,余額不足!");
// }
}
public static void main(String[] args){
Account acct = new Account("123456",1000);
new DrawThread("A",acct,800).start();
new DrawThread("B",acct,800).start();
}
}
上面代碼中注釋掉的部分:(1)synchronized同步代碼塊 (2)線程休眠。如果注釋掉(1)、(2),則運行結(jié)果有多種可能性,可能性之一(概率較?。险_壿嫞?br />
B取錢成功,吐出鈔票:800.0
余額為 : 200.0
A取錢失敗,余額不足!
應(yīng)該是B先強找到取錢資源,并且正確修改余額后,A才開始判斷用戶余額;這種概率非常小,多數(shù)運行會類似以下情況:
A取錢成功,吐出鈔票:800.0
B取錢成功,吐出鈔票:800.0
余額為 : -600.0
余額為 : 200.0
這明顯是不符合邏輯的,從運行結(jié)果可以猜測,A先搶占資源,取出金額,但在修改余額之前,資源被B搶占;由于余額未被修改,則B看到余額仍然是800,B仍然取出金額;A先運行修改余額,但并未打印,B搶奪資源;B修改余額,并打印余額,為-600;A打印余額,為200;
如果加上(2)線程休眠,則一定是錯誤狀況,因為A或B在取出金額后一定會因為sleep釋放CPU資源,JVM會調(diào)用其他處于準備狀態(tài)的進程。第二個取錢判斷余額一定是錯誤的。
如果加上(1)synchronized同步代碼塊,在線程run方法體中對account進行加鎖;則每次都會保證執(zhí)行邏輯正常:
A取錢成功,吐出鈔票:800.0
余額為 : 200.0
B取錢失敗,余額不足!
可以設(shè)想一下執(zhí)行過程:
A先搶占資源,在run方法體初始對account類進行加鎖;然后開始執(zhí)行同步代碼塊;如果執(zhí)行到中間某個環(huán)節(jié),CPU資源被B搶占;B開始執(zhí)行,一開始也會對account類進行加鎖。但是加鎖時會發(fā)現(xiàn)account已經(jīng)被A占用,則會調(diào)整為阻塞狀態(tài)等待A釋放資源;A執(zhí)行完同步代碼塊后釋放account的鎖,B繼續(xù)執(zhí)行;B運行時看到的余額保證是A已經(jīng)修改過的,會按照正確邏輯正常執(zhí)行。
總結(jié)
以上就是本文關(guān)于java編程多線程并發(fā)處理實例解析的全部內(nèi)容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
欄 目:Java編程
本文地址:http://www.jygsgssxh.com/a1/Javabiancheng/8350.html
您可能感興趣的文章
- 01-10Java咖啡館(1)——嘆咖啡
- 01-10Java Socket編程(三) 服務(wù)器Sockets
- 01-10Java進階:Struts多模塊的技巧
- 01-10Java Socket編程(一) Socket傳輸模式
- 01-10Java Socket編程(二) Java面向連接的類
- 01-10Java運行時多態(tài)性的實現(xiàn)
- 01-10Java經(jīng)驗點滴:處理沒有被捕獲的異常
- 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)器
- 01-10Java中的浮點數(shù)分析
- 01-10面向?qū)ο缶幊?Java中的抽象數(shù)據(jù)類型


閱讀排行
本欄相關(guān)
- 01-10Java咖啡館(1)——嘆咖啡
- 01-10JVM的垃圾回收機制詳解和調(diào)優(yōu)
- 01-10Java Socket編程(三) 服務(wù)器Sockets
- 01-10Java進階:Struts多模塊的技巧
- 01-10J2SE 1.5版本的新特性一覽
- 01-10Java Socket編程(一) Socket傳輸模式
- 01-10Java運行時多態(tài)性的實現(xiàn)
- 01-10Java Socket編程(二) Java面向連接的類
- 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)
- 01-10Java經(jīng)驗點滴:處理沒有被捕獲的異常
隨機閱讀
- 01-11ajax實現(xiàn)頁面的局部加載
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實例總結(jié)
- 04-02jquery與jsp,用jquery
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個骰子


