Java synchronized線程交替運(yùn)行實(shí)現(xiàn)過程詳解
背景
用兩個線程交替輸出A-Z和1-26,即一個線程輸出A-Z,另一個線程輸出1-26
而且是交替形式
- 線程1輸出A——線程二輸出1
- 線程1輸出B——線程二輸出2
- 線程1輸出C——線程二輸出3
以此類推
分析
主要考察線程之間的通信,思路就是創(chuàng)建兩個線程
在一個線程輸出一個內(nèi)容之后,自己進(jìn)入阻塞,去喚醒另一個線程
另一個線程同樣,輸出一個內(nèi)容之后,自己進(jìn)入阻塞,去喚醒另一個線程
代碼實(shí)現(xiàn)(一)
public class AlternateCover {
public static void main(String[] args) {
final char[] arrLetter = "ABCDEFGHIJKLMNOPQRSTUVWXYZ".toCharArray();
final String[] arrNumber = {"1","2","3","4","5","6","7","8","9","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26"};
threadRun(arrLetter, arrNumber);
}
private static void threadRun(char[] arrLetter,String[] arrNumber){
final Object lock = new Object();// 設(shè)置一個鎖對象
// print arrNumber
new Thread(() -> {
synchronized (lock) {
for (String a : arrNumber) {
System.out.print( a);
try {
lock.notify();// 喚醒其他等待的線程 此處喚醒 arrLetter
lock.wait();// arrNumber自己進(jìn)入等待 讓出CPU資源和鎖資源
} catch (InterruptedException e) {
e.printStackTrace();
}
}
lock.notify();
}
}, "arrNumber ").start();
// print arrLetter
new Thread(() -> {
synchronized (lock) {// 獲取對象鎖
for (char a : arrLetter) {
System.out.print(a);
try {
lock.notify();// 喚醒其他等待的線程 此處喚醒 arrNumber
lock.wait();// arrLetter自己進(jìn)入等待 讓出CPU資源和鎖資源
} catch (InterruptedException e) {
e.printStackTrace();
}
}
lock.notify();// 最后那個等待的線程需要被喚醒,否則程序無法結(jié)束
}
}, "arrLetter ").start();
}
}
運(yùn)行一下,確實(shí)實(shí)現(xiàn)了交替輸出,但是多運(yùn)行幾次,就會發(fā)現(xiàn)問題
有時候是數(shù)字先輸出,有時候是字母先輸出
即兩個線程誰先啟動的順序是不固定的
倘若試題中再加一句,必須要字母先輸出,怎么辦?
代碼實(shí)現(xiàn)(二)
/**
* 交替掩護(hù) 必須保證大寫字母先輸出
*/
public class AlternateCover {
public static volatile Boolean flg = false;// 誰先開始的標(biāo)志 volatile修飾目的是讓該值修改對所有線程可見,且防止指令重排序
public static void main(String[] args) {
final char[] arrLetter = "ABCDEFGHIJKLMNOPQRSTUVWXYZ".toCharArray();
final String[] arrNumber = {"1","2","3","4","5","6","7","8","9","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26"};
threadRun(arrLetter, arrNumber);
}
private static void threadRun(char[] arrLetter,String[] arrNumber){
final Object lock = new Object();// 鎖對象
// print arrLetter
new Thread(() -> {
synchronized (lock) {
if (!flg){ // 如果flg是false 就將值設(shè)為true
flg = true;
}
for (char a : arrLetter) {
System.out.print(a);// 輸出內(nèi)容
try {
lock.notify();// 喚醒在等待的其他線程中的一個(此處也只有另一個)
lock.wait();// 自己進(jìn)入等待 讓出CPU資源和鎖資源
} catch (InterruptedException e) {
e.printStackTrace();
}
}
lock.notify();// 最后那個等待的線程需要被喚醒,否則程序無法結(jié)束
}
}, "arrLetter").start();
// print arrNumber
new Thread(() -> {
synchronized (lock) {
if (!flg){// 倘若是該線程先執(zhí)行,那么flg次數(shù)還是false 就先等著
try {
lock.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
for (String a : arrNumber) {
System.out.print( a);
try {
lock.notify();
lock.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
lock.notify();
}
}, "arrNumber").start();
}
}
如此問題可以得到解決,但有更優(yōu)(裝)雅(B)的解決辦法
CountDownLatch實(shí)現(xiàn)
/**
* 交替掩護(hù) 必須保證大寫字母先輸出
*/
public class AlternateCover {
private static CountDownLatch count = new CountDownLatch(1);// 計(jì)數(shù)器容量為1
public static void main(String[] args) {
final char[] arrLetter = "ABCDEFGHIJKLMNOPQRSTUVWXYZ".toCharArray();
final String[] arrNumber = {"1","2","3","4","5","6","7","8","9","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26"};
threadRun(arrLetter, arrNumber);
}
private static void threadRun(char[] arrLetter,String[] arrNumber){
final Object lock = new Object();
// print arrLetter
new Thread(() -> {
synchronized (lock) {// 獲取對象鎖
count.countDown();// 對計(jì)數(shù)器進(jìn)行遞減1操作,當(dāng)計(jì)數(shù)器遞減至0時,當(dāng)前線程會去喚醒阻塞隊(duì)列里的所有線程(只針對count)
for (char a : arrLetter) {
System.out.print(a);
try {
lock.notify();// 喚醒其他等待的線程 此處喚醒 arrNumber
lock.wait();// arrLetter自己進(jìn)入等待 讓出CPU資源和鎖資源
} catch (InterruptedException e) {
e.printStackTrace();
}
}
lock.notify();// 最后那個等待的線程需要被喚醒,否則程序無法結(jié)束
}
}, "arrLetter ").start();
// print arrNumber
new Thread(() -> {
synchronized (lock) {
try {
count.await();// 如果該線程先執(zhí)行 阻塞當(dāng)前線程,將當(dāng)前線程加入阻塞隊(duì)列
} catch (InterruptedException e) {
e.printStackTrace();
}
for (String a : arrNumber) {
System.out.print( a);
try {
lock.notify();// 喚醒其他等待的線程 此處喚醒 arrLetter
lock.wait();// arrNumber自己進(jìn)入等待 讓出CPU資源和鎖資源
} catch (InterruptedException e) {
e.printStackTrace();
}
}
lock.notify();
}
}, "arrNumber ").start();
}
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Servlet+MyBatis項(xiàng)目轉(zhuǎn)Spring Cloud微服務(wù),多數(shù)據(jù)源配置修改建議
今天小編就為大家分享一篇關(guān)于Servlet+MyBatis項(xiàng)目轉(zhuǎn)Spring Cloud微服務(wù),多數(shù)據(jù)源配置修改建議,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-01-01
深入理解Java基礎(chǔ)之try-with-resource語法糖
這篇文章主要介紹了深入理解Java基礎(chǔ)之try-with-resource語法糖,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-02-02
Spring boot工具類靜態(tài)屬性注入及多環(huán)境配置詳解
這篇文章主要為大家詳細(xì)介紹了Spring boot工具類靜態(tài)屬性注入,及多環(huán)境配置詳解,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-04-04
mybatis Plus 多表聯(lián)合查詢的實(shí)現(xiàn)示例
這篇文章主要介紹了mybatis Plus 多表聯(lián)合查詢的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
Java 如何將 byte[] 轉(zhuǎn)換為 File 對象
本文介紹了如何將字節(jié)數(shù)組(byte[])轉(zhuǎn)換為文件對象(File)的方法,提供了具體代碼示例,代碼簡單易懂,感興趣的朋友跟隨小編一起看看吧2025-03-03
Springboot通過run啟動web應(yīng)用的方法
這篇文章主要介紹了Springboot通過run啟動web應(yīng)用的方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-03-03
idea項(xiàng)目中target文件提示拒絕訪問的解決
這篇文章主要介紹了idea項(xiàng)目中target文件提示拒絕訪問的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-11-11

