Java并發(fā)編程線程間通訊實(shí)現(xiàn)過(guò)程詳解
在Java中線程間通訊有多種方式,我這里列出一些常用方式,并用代碼的方式展示他們是如何實(shí)現(xiàn)的:
- 共享變量
- wait, notify,notifyAll(這3個(gè)方法是Object對(duì)象中的方法,且必須與synchronized關(guān)鍵字結(jié)合使用)
- CyclicBarrier、CountDownLatch
- 利用LockSupport
- Lock/Condition機(jī)制
- 管道,創(chuàng)建管道輸出流PipedOutputStream和管道輸入流PipedInputStream
示例一:
package com.zhi.test;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger;
import org.junit.Test;
/**
* Java多線程-線程通訊示例<br>
* flag作為共享變量JobB執(zhí)行,notify通知Job執(zhí)行,CountDownLatch通知主線程執(zhí)行
*
* @author 張遠(yuǎn)志
* @since 2020年5月4日21:51:24
*
*/
public class ThreadTest2 {
private CountDownLatch latch;
private volatile boolean flag = true;
private Object lock = new Object();
private AtomicInteger num = new AtomicInteger(0);
class JobA implements Runnable {
@Override
public void run() {
synchronized (lock) {
flag = false;
if (num.get() != 3) {
try {
lock.wait(); // wait方法會(huì)釋放鎖
} catch (InterruptedException e) {
}
}
System.out.println("任務(wù)A收到通知,繼續(xù)執(zhí)行作業(yè)");
}
latch.countDown();
}
}
class JobB implements Runnable {
@Override
public void run() {
while (flag) { // 保證JobA先申請(qǐng)到鎖
}
synchronized (lock) {
for (int i = 1; i <= 5; i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
}
int a = num.incrementAndGet();
System.out.println("任務(wù)B第" + i + "次執(zhí)行,num值為:" + a);
if (a == 3) {
lock.notify(); // 喚醒JobB線程,notify方法不會(huì)釋放鎖
}
}
}
latch.countDown();
}
}
@Test
public void test() {
latch = new CountDownLatch(2);
new Thread(new JobA()).start();
new Thread(new JobB()).start();
try {
latch.await(); // 保證2個(gè)線程都執(zhí)行完畢
} catch (InterruptedException e) {
}
}
}
結(jié)果輸出:
任務(wù)B第1次執(zhí)行,num值為:1
任務(wù)B第2次執(zhí)行,num值為:2
任務(wù)B第3次執(zhí)行,num值為:3
任務(wù)B第4次執(zhí)行,num值為:4
任務(wù)B第5次執(zhí)行,num值為:5
任務(wù)A收到通知,繼續(xù)執(zhí)行作業(yè)
示例二:
package com.zhi.test;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.locks.LockSupport;
import org.junit.Test;
/**
* Java多線程-線程通訊示例,利用LockSupport
*
* @author 張遠(yuǎn)志
* @since 2020年5月4日21:51:24
*
*/
public class ThreadTest3 {
private CountDownLatch latch;
private volatile int num = 0;
private Thread ta;
private Thread tb;
class JobA implements Runnable {
@Override
public void run() {
if (num != 3) {
LockSupport.park();
}
System.out.println("任務(wù)A收到通知,繼續(xù)執(zhí)行作業(yè)");
latch.countDown();
}
}
class JobB implements Runnable {
@Override
public void run() {
for (int i = 1; i <= 5; i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
}
num++;
System.out.println("任務(wù)B第" + i + "次執(zhí)行,num值為:" + num);
if (num == 3) {
LockSupport.unpark(ta); // unpark會(huì)立即激活傳入線程
}
}
latch.countDown();
}
}
@Test
public void test() {
latch = new CountDownLatch(2);
ta = new Thread(new JobA());
tb = new Thread(new JobB());
ta.start();
tb.start();
try {
latch.await(); // 保證2個(gè)線程都執(zhí)行完畢
} catch (InterruptedException e) {
}
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java學(xué)生信息管理系統(tǒng)設(shè)計(jì)(數(shù)據(jù)庫(kù)版)
這篇文章主要為大家詳細(xì)介紹了數(shù)據(jù)庫(kù)版的Java學(xué)生信息管理系統(tǒng)設(shè)計(jì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
導(dǎo)入maven項(xiàng)目各個(gè)注解均報(bào)錯(cuò)的解決方案
這篇文章主要介紹了導(dǎo)入maven項(xiàng)目各個(gè)注解均報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12
SpringMVC 使用JSR-303進(jìn)行校驗(yàn) @Valid示例
本篇文章主要介紹了SpringMVC 使用JSR-303進(jìn)行校驗(yàn) @Valid示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02
spring的TransactionSynchronizationAdapter事務(wù)源碼解析
這篇文章主要介紹了spring的TransactionSynchronizationAdapter事務(wù)源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
java利用正則表達(dá)式處理特殊字符的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于java利用正則表達(dá)式處理特殊字符的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12

