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é)果有多種可能性,可能性之一(概率較小),符合正常邏輯:
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)用其他處于準(zhǔn)備狀態(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網(wǎng)絡(luò)編程和多線程的多對多聊天系統(tǒng)
- 教你如何使用Java多線程編程LockSupport工具類
- Java多線程實現(xiàn)TCP網(wǎng)絡(luò)Socket編程(C/S通信)
- 淺談java多線程編程
- Java多線程并發(fā)編程和鎖原理解析
- JAVA多線程編程實例詳解
- 深入理解Java多線程與并發(fā)編程
- Java多線程編程實戰(zhàn)之模擬大量數(shù)據(jù)同步
- Java編程一道多線程問題實例代碼
- Java編程多線程之共享數(shù)據(jù)代碼詳解
- 關(guān)于Java多線程編程鎖優(yōu)化的深入學(xué)習(xí)
- java多線程編程同步器Future和FutureTask解析及代碼示例
- java多線程編程實例
- Java編程之多線程死鎖與線程間通信簡單實現(xiàn)代碼
- Java進階必備之多線程編程
相關(guān)文章
maven繼承父工程統(tǒng)一版本號的實現(xiàn)
這篇文章主要介紹了maven繼承父工程統(tǒng)一版本號的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08Java不借助第三變量實現(xiàn)兩數(shù)交換的示例
本文主要介紹了Java不借助第三變量實現(xiàn)兩數(shù)交換的示例,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02五種SpringBoot實現(xiàn)數(shù)據(jù)加密存儲的方式總結(jié)
這篇文章主要為大家詳細介紹了五種常見數(shù)據(jù)加密存儲的方法(結(jié)合SpringBoot和MyBatisPlus框架進行實現(xiàn)),文中的示例代碼講解詳細,需要的可以參考下2023-11-11Springboot?中的?Filter?實現(xiàn)超大響應(yīng)?JSON?數(shù)據(jù)壓縮的方法
這篇文章主要介紹了Springboot?中的?Filter?實現(xiàn)超大響應(yīng)?JSON?數(shù)據(jù)壓縮,定義GzipFilter對輸出進行攔截,定義 Controller該 Controller 非常簡單,主要讀取一個大文本文件,作為輸出的內(nèi)容,本文通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下2022-10-10