Java使用鎖解決銀行取錢問題實例分析
本文實例講述了Java使用鎖解決銀行取錢問題。分享給大家供大家參考,具體如下:
一 點睛
1 釋放同步監(jiān)視器
線程會在如下幾種情況下釋放對同步監(jiān)視器的鎖定:
- 當前線程的同步方法、同步代碼塊執(zhí)行結(jié)束,當前線程即釋放同步監(jiān)視器。
- 當線程在同步代碼塊、同步方法中遇到break、return終止了該代碼塊、該方法的繼續(xù)執(zhí)行,當前線程將會釋放同步監(jiān)視器。
- 當線程在同步代碼塊、同步方法中出現(xiàn)了未處理的Error或Exception,導致了該代碼塊、該方法異常結(jié)束時將會釋放同步監(jiān)視器。
- 當線程執(zhí)行同步代碼塊或同步方法時,程序執(zhí)行了同步監(jiān)視器對象的wait()方法,則當前線程暫停,并釋放同步監(jiān)視器。
2 同步鎖
Lock是控制多個線程對共享資源進行訪問的工具。
通常,鎖提供了對共享資源的獨占訪問,每次只能有一個線程對Lock對象加鎖,線程開始訪問共享資源之前應先獲得Lock對象。
不過,某些鎖可能允許對共享資源并發(fā)訪問,如 ReadWriteLock(讀寫鎖)。當然,在實現(xiàn)線程安全的控制中,通常喜歡使用ReentrantLock(可重入鎖)。使用該Lock對象可以顯式地加鎖、釋放鎖。
ReentrantLock鎖具有可重入性,也就是說線程可以對已加鎖的ReentrantLock鎖再次加鎖,ReentrantLock對象會維持一個計數(shù)器來追蹤lock方法的嵌套調(diào)用,線程在每次調(diào)用lock()方法加鎖后,必須顯式調(diào)用unlock()方法來釋放鎖,所以一段被鎖保護的代碼可以調(diào)用另一個被相同鎖保護的方法。
二 代碼
1 定義一個賬戶類
import java.util.concurrent.locks.*; public class Account { // 定義鎖對象 private final ReentrantLock lock = new ReentrantLock(); // 封裝賬戶編號、賬戶余額的兩個成員變量 private String accountNo; private double balance; public Account(){} // 構(gòu)造器 public Account(String accountNo , double balance) { this.accountNo = accountNo; this.balance = balance; } // accountNo的setter和getter方法 public void setAccountNo(String accountNo) { this.accountNo = accountNo; } public String getAccountNo() { return this.accountNo; } // 因此賬戶余額不允許隨便修改,所以只為balance提供getter方法, public double getBalance() { return this.balance; } // 提供一個線程安全draw()方法來完成取錢操作 public void draw(double drawAmount) { // 加鎖 lock.lock(); try { // 賬戶余額大于取錢數(shù)目 if (balance >= drawAmount) { // 吐出鈔票 System.out.println(Thread.currentThread().getName() + "取錢成功!吐出鈔票:" + drawAmount); try { Thread.sleep(1); } catch (InterruptedException ex) { ex.printStackTrace(); } // 修改余額 balance -= drawAmount; System.out.println("\t余額為: " + balance); } else { System.out.println(Thread.currentThread().getName() + "取錢失??!余額不足!"); } } finally { // 修改完成,釋放鎖 lock.unlock(); } } // 下面兩個方法根據(jù)accountNo來重寫hashCode()和equals()方法 public int hashCode() { return accountNo.hashCode(); } 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; } }
2 定義一個取錢線程
public class DrawThread extends Thread { // 模擬用戶賬戶 private Account account; // 當前取錢線程所希望取的錢數(shù) private double drawAmount; public DrawThread(String name , Account account , double drawAmount) { super(name); this.account = account; this.drawAmount = drawAmount; } // 當多條線程修改同一個共享數(shù)據(jù)時,將涉及數(shù)據(jù)安全問題。 public void run() { // 直接調(diào)用account對象的draw方法來執(zhí)行取錢 // 同步方法的同步監(jiān)視器是this,this代表調(diào)用draw()方法的對象。 // 也就是說:線程進入draw()方法之前,必須先對account對象的加鎖。 account.draw(drawAmount); } }
3 測試主類
public class DrawTest { public static void main(String[] args) { // 創(chuàng)建一個賬戶 Account acct = new Account("1234567" , 1000); // 模擬兩個線程對同一個賬戶取錢 new DrawThread("甲" , acct , 800).start(); new DrawThread("乙" , acct , 800).start(); } }
三 運行
乙取錢成功!吐出鈔票:800.0
余額為: 200.0
甲取錢失??!余額不足!
四 說明
Account類中定義了一個ReentrantLock對象,程序?qū)崿F(xiàn)draw()方法時,進入方法開始執(zhí)行后立即請求對ReentrantLock對象進行加鎖,當執(zhí)行完draw()方法的取錢邏輯之后,程序使用finally塊來確保釋放鎖。
更多java相關內(nèi)容感興趣的讀者可查看本站專題:《Java進程與線程操作技巧總結(jié)》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對大家java程序設計有所幫助。
相關文章
Java實現(xiàn)單鏈表反轉(zhuǎn)的多種方法總結(jié)
這篇文章主要給大家介紹了關于Java實現(xiàn)單鏈表反轉(zhuǎn)的多種方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-04-04在同一個類中調(diào)用帶有@Transactional注解的方法示例
這篇文章主要為大家介紹了在同一個類中調(diào)用帶有@Transactional注解的方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-04-04PowerJob的DispatchStrategy方法工作流程源碼解讀
這篇文章主要為大家介紹了PowerJob的DispatchStrategy方法工作流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01Intellij IDEA中啟動多個微服務(開啟Run Dashboard管理)
這篇文章主要介紹了Intellij IDEA中啟動多個微服務(開啟Run Dashboard管理),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-07-07解析Spring Boot 如何讓你的 bean 在其他 bean&n
在 SpringBoot 中如何讓自己的某個指定的 Bean 在其他 Bean 前完成被 Spring 加載?我聽到這個問題的第一反應是,為什么會有這樣奇怪的需求?下面小編給大家分析下Spring Boot 如何讓你的 bean 在其他 bean 之前完成加載 ,感興趣的朋友一起看看吧2024-01-01不寫mybatis的@Param有的報錯有的卻不報錯問題分析
這篇文章主要為大家介紹了不寫mybatis的@Param有的報錯有的卻不報錯問題分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09