Java線程協(xié)調(diào)運(yùn)行操作實(shí)例詳解
本文實(shí)例講述了Java線程協(xié)調(diào)運(yùn)行操作。分享給大家供大家參考,具體如下:
一 點(diǎn)睛
借助于Object類提供的wait()、notify()和notifyAll()三個(gè)方法,可實(shí)現(xiàn)Java線程協(xié)調(diào)運(yùn)行。這三個(gè)方法并不屬于Thread類,而是屬于Object類。但這三個(gè)方法必須同步監(jiān)視器對(duì)象調(diào)用。
關(guān)于這三個(gè)方法的解釋如下:
- wait():導(dǎo)致當(dāng)前線程等待,直到其他線程調(diào)用該同步監(jiān)視器的notify()方法或notifyAll()方法來(lái)喚醒該線程。該wait()方法有三種形式:無(wú)時(shí)間參數(shù)的wait(一直等待,直到其他線程通知),帶毫秒?yún)?shù)的wait和帶毫秒、微秒?yún)?shù)的wait(這兩種方法都是等待指定時(shí)間后自動(dòng)蘇醒)。調(diào)用wait()方法的當(dāng)前線程會(huì)釋放對(duì)該同步監(jiān)視器的鎖定。
- notify():?jiǎn)拘言诖送奖O(jiān)視器上等待的單個(gè)線程。如果所有線程都在此同步監(jiān)視器上等待,則會(huì)選擇喚醒其中一個(gè)線程。選擇是任意性的。只有當(dāng)前線程放棄對(duì)該同步監(jiān)視器的鎖定后(使用wait()方法),才可以執(zhí)行被喚醒的線程。
- notifyAll():?jiǎn)拘言诖送奖O(jiān)視器上等待的所有線程。只有當(dāng)前線程放棄對(duì)該同步監(jiān)視器的鎖定后,才可以執(zhí)行被喚醒的線程。
對(duì)于使用synchronized修飾的同步方法,因?yàn)樵擃惖哪J(rèn)實(shí)例(this)就是同步監(jiān)視器,所以可以直接調(diào)用這三個(gè)方法。
對(duì)于使用synchronized修飾的同步塊,同步監(jiān)視器是synchronized后括號(hào)里的對(duì)象,所以必須使用該對(duì)象調(diào)用這三個(gè)方法。
二 實(shí)戰(zhàn)
1 Account類
public class Account
{
// 封裝賬戶編號(hào)、賬戶余額的兩個(gè)成員變量
private String accountNo;
private double balance;
// 標(biāo)識(shí)賬戶中是否已有存款的旗標(biāo)
private boolean flag = false;
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;
}
public synchronized void draw(double drawAmount)
{
try
{
// 如果flag為假,表明賬戶中還沒(méi)有人存錢進(jìn)去,取錢方法阻塞
if (!flag)
{
wait();
}
else
{
// 執(zhí)行取錢
System.out.println(Thread.currentThread().getName()
+ " 取錢:" + drawAmount);
balance -= drawAmount;
System.out.println("賬戶余額為:" + balance);
// 將標(biāo)識(shí)賬戶是否已有存款的旗標(biāo)設(shè)為false。
flag = false;
// 喚醒其他線程
notifyAll();
}
}
catch (InterruptedException ex)
{
ex.printStackTrace();
}
}
public synchronized void deposit(double depositAmount)
{
try
{
// 如果flag為真,表明賬戶中已有人存錢進(jìn)去,則存錢方法阻塞
if (flag) //①
{
wait();
}
else
{
// 執(zhí)行存款
System.out.println(Thread.currentThread().getName()
+ " 存款:" + depositAmount);
balance += depositAmount;
System.out.println("賬戶余額為:" + balance);
// 將表示賬戶是否已有存款的旗標(biāo)設(shè)為true
flag = true;
// 喚醒其他線程
notifyAll();
}
}
catch (InterruptedException ex)
{
ex.printStackTrace();
}
}
// 下面兩個(gè)方法根據(jù)accountNo來(lái)重寫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 DrawThread線程類
public class DrawThread extends Thread
{
// 模擬用戶賬戶
private Account account;
// 當(dāng)前取錢線程所希望取的錢數(shù)
private double drawAmount;
public DrawThread(String name , Account account
, double drawAmount)
{
super(name);
this.account = account;
this.drawAmount = drawAmount;
}
// 重復(fù)100次執(zhí)行取錢操作
public void run()
{
for (int i = 0 ; i < 100 ; i++ )
{
account.draw(drawAmount);
}
}
}
3 DepositThread線程類
public class DepositThread extends Thread
{
// 模擬用戶賬戶
private Account account;
// 當(dāng)前取錢線程所希望存款的錢數(shù)
private double depositAmount;
public DepositThread(String name , Account account
, double depositAmount)
{
super(name);
this.account = account;
this.depositAmount = depositAmount;
}
// 重復(fù)100次執(zhí)行存款操作
public void run()
{
for (int i = 0 ; i < 100 ; i++ )
{
account.deposit(depositAmount);
}
}
}
4 測(cè)試類
public class DrawTest
{
public static void main(String[] args)
{
// 創(chuàng)建一個(gè)賬戶
Account acct = new Account("1234567" , 0);
new DrawThread("取錢者" , acct , 800).start();
new DepositThread("存款者甲" , acct , 800).start();
new DepositThread("存款者乙" , acct , 800).start();
new DepositThread("存款者丙" , acct , 800).start();
}
}
三 運(yùn)行結(jié)果
......
存款者甲 存款:800.0
賬戶余額為:800.0
取錢者 取錢:800.0
賬戶余額為:0.0
存款者丙 存款:800.0
賬戶余額為:800.0
取錢者 取錢:800.0
賬戶余額為:0.0
存款者甲 存款:800.0
賬戶余額為:800.0
四 說(shuō)明
運(yùn)行該程序,可以看到存款者線程、取錢者線程交替執(zhí)行的情形,每當(dāng)存款者向賬戶中存入800元之后,取錢這線程立即從賬戶中取出這筆錢。存款完成后賬戶余額總是800元,取錢結(jié)束后賬戶余額總是0元。
三個(gè)存款者線程隨機(jī)地向賬戶中存款,只有一個(gè)取錢者線程執(zhí)行取錢操作。只有當(dāng)取錢者取錢后,存款者才可以存款;同理,只有存款者存款后,取錢這線程才可以取錢。
程序最后被阻塞無(wú)法繼續(xù)向下執(zhí)行,這是因?yàn)?個(gè)存款者線程共300次存款操作,但1個(gè)取錢者線程只有100次操作,所以程序最后被阻塞。
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java進(jìn)程與線程操作技巧總結(jié)》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點(diǎn)技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對(duì)大家java程序設(shè)計(jì)有所幫助。
- Java后臺(tái)線程操作示例【守護(hù)線程】
- Java中join線程操作實(shí)例分析
- Java使用Callable和Future創(chuàng)建線程操作示例
- 詳解Java多線程編程中線程的啟動(dòng)、中斷或終止操作
- Java線程的start方法回調(diào)run方法的操作技巧
- Java多線程編程中使用Condition類操作鎖的方法詳解
- Java8并行流中自定義線程池操作示例
- Java使用Thread創(chuàng)建多線程并啟動(dòng)操作示例
- java線程同步操作實(shí)例詳解
- Java線程操作的常見(jiàn)方法【線程名稱獲取、設(shè)置、線程啟動(dòng)判斷等】
相關(guān)文章
在springboot3微項(xiàng)目中如何用idea批量創(chuàng)建單元測(cè)試邏輯
這篇文章主要介紹了在SpringBoot3項(xiàng)目中使用IntelliJIDEA批量創(chuàng)建單元測(cè)試包括準(zhǔn)備工作(確保項(xiàng)目配置正確,添加測(cè)試依賴),使用IntelliJIDEA創(chuàng)建測(cè)試,感興趣的朋友一起看看吧2024-10-10
詳解如何配置Spring Batch批處理失敗重試機(jī)制
這篇文章主要來(lái)和大家一起探討一下如何在Spring批處理框架中配置重試邏輯,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-06-06
基于SpringBoot和Hutool工具包實(shí)現(xiàn)驗(yàn)證碼的案例
隨著安全性的要求越來(lái)越高,目前項(xiàng)目中很多都會(huì)使用驗(yàn)證碼,只要涉及到登錄,絕大多數(shù)都會(huì)有驗(yàn)證的要求,驗(yàn)證碼的形式也是多種多樣,更復(fù)雜的圖形驗(yàn)證碼和行為驗(yàn)證碼已經(jīng)成為了更流行的趨勢(shì),本文給大家介紹了SpringBoot Hutool實(shí)現(xiàn)驗(yàn)證碼的案例,需要的朋友可以參考下2024-05-05
spring?java?動(dòng)態(tài)獲取consul?K/V的方法
這篇文章主要介紹了spring?java?動(dòng)態(tài)獲取consul?K/V的相關(guān)資料,主要包括springConsul配置kv路徑以及自動(dòng)注入consulKV到服務(wù)中,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-10-10
Knife4j?3.0.3?整合SpringBoot?2.6.4的詳細(xì)過(guò)程
本文要講的是?Knife4j?3.0.3?整合SpringBoot?2.6.4,在SpringBoot?2.4以上的版本和之前的版本還是不一樣的,這個(gè)也容易導(dǎo)致一些問(wèn)題,本文就這兩個(gè)版本的整合做一個(gè)實(shí)戰(zhàn)介紹2022-09-09
基于Flyway實(shí)現(xiàn)簡(jiǎn)化Spring Boot項(xiàng)目部署
這篇文章主要介紹了基于Flyway實(shí)現(xiàn)簡(jiǎn)化Spring Boot項(xiàng)目部署,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06

