网易首页 > 网易号 > 正文 申请入驻

面试官问:Java线程中如何保证通信,都有哪些方式?

0
分享至

一、介绍
二、线程间的通信方式①同步

这里讲的同步是指多个线程通过synchronized关键字这种方式来实现线程间的通信。

参考示例:

public class MyObject {

synchronized public void methodA() {
//do something....
}

synchronized public void methodB() {
//do some other thing
}
}

public class ThreadA extends Thread {

private MyObject object;
//省略构造方法
@Override
public void run() {
super.run();
object.methodA();
}
}

public class ThreadB extends Thread {

private MyObject object;
//省略构造方法
@Override
public void run() {
super.run();
object.methodB();
}
}

public class Run {
public static void main(String[] args) {
MyObject object = new MyObject();

//线程A与线程B 持有的是同一个对象:object
ThreadA a = new ThreadA(object);
ThreadB b = new ThreadB(object);
a.start();
b.start();
}
}

由于线程A和线程B持有同一个MyObject类的对象object,尽管这两个线程需要调用不同的方法,但是它们是同步执行的,比如:线程B需要等待线程A执行完了methodA()方法之后,它才能执行methodB()方法。这样,线程A和线程B就实现了 通信。

这种方式,本质上就是“共享内存”式的通信。多个线程需要访问同一个共享变量,谁拿到了锁(获得了访问权限),谁就可以执行。

②while轮询的方式

代码如下:

import java.util.ArrayList;
import java.util.List;

public class MyList {

private List list = new ArrayList();
public void add() {
list.add("elements");
}
public int size() {
return list.size();
}
}

import mylist.MyList;

public class ThreadA extends Thread {

private MyList list;

public ThreadA(MyList list) {
super();
this.list = list;
}

@Override
public void run() {
try {
for (int i = 0; i < 10; i++) {
list.add();
System.out.println("添加了" + (i + 1) + "个元素");
Thread.sleep(1000);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

import mylist.MyList;

public class ThreadB extends Thread {

private MyList list;

public ThreadB(MyList list) {
super();
this.list = list;
}

@Override
public void run() {
try {
while (true) {
if (list.size() == 5) {
System.out.println("==5, 线程b准备退出了");
throw new InterruptedException();
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

import mylist.MyList;
import extthread.ThreadA;
import extthread.ThreadB;

public class Test {

public static void main(String[] args) {
MyList service = new MyList();

ThreadA a = new ThreadA(service);
a.setName("A");
a.start();

ThreadB b = new ThreadB(service);
b.setName("B");
b.start();
}
}

在这种方式下,线程A不断地改变条件,线程ThreadB不停地通过while语句检测这个条件(list.size()==5)是否成立 ,从而实现了线程间的通信。但是这种方式会浪费CPU资源。

之所以说它浪费资源,是因为JVM调度器将CPU交给线程B执行时,它没做啥“有用”的工作,只是在不断地测试 某个条件是否成立。就类似于现实生活中,某个人一直看着手机屏幕是否有电话来了,而不是:在干别的事情,当有电话来时,响铃通知TA电话来了。

这种方式还存在另外一个问题:

轮询的条件的可见性问题,关于内存可见性问题,可参考:JAVA多线程之volatile 与 synchronized 的比较中的第一点“一,volatile关键字的可见性”

http://www.cnblogs.com/hapjin/p/5492880.html

线程都是先把变量读取到本地线程栈空间,然后再去再去修改的本地变量。因此,如果线程B每次都在取本地的 条件变量,那么尽管另外一个线程已经改变了轮询的条件,它也察觉不到,这样也会造成死循环。

③wait/notify机制

代码如下:

import java.util.ArrayList;
import java.util.List;

public class MyList {

private static List list = new ArrayList();

public static void add() {
list.add("anyString");
}

public static int size() {
return list.size();
}
}

public class ThreadA extends Thread {

private Object lock;

public ThreadA(Object lock) {
super();
this.lock = lock;
}

@Override
public void run() {
try {
synchronized (lock) {
if (MyList.size() != 5) {
System.out.println("wait begin "
+ System.currentTimeMillis());
lock.wait();
System.out.println("wait end "
+ System.currentTimeMillis());
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

public class ThreadB extends Thread {
private Object lock;

public ThreadB(Object lock) {
super();
this.lock = lock;
}

@Override
public void run() {
try {
synchronized (lock) {
for (int i = 0; i < 10; i++) {
MyList.add();
if (MyList.size() == 5) {
lock.notify();
System.out.println("已经发出了通知");
}
System.out.println("添加了" + (i + 1) + "个元素!");
Thread.sleep(1000);
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

public class Run {

public static void main(String[] args) {

try {
Object lock = new Object();

ThreadA a = new ThreadA(lock);
a.start();

Thread.sleep(50);

ThreadB b = new ThreadB(lock);
b.start();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

线程A要等待某个条件满足时(list.size()==5),才执行操作。线程B则向list中添加元素,改变list 的size。更多面试资料,公众 号Java精选,回复java面试,获取面试资料,支持在线刷题。

A,B之间如何通信的呢?也就是说,线程A如何知道 list.size() 已经为5了呢?

这里用到了Object类的 wait() 和 notify() 方法。

当条件未满足时(list.size() !=5),线程A调用wait() 放弃CPU,并进入阻塞状态。---不像②while轮询那样占用CPU

当条件满足时,线程B调用 notify()通知 线程A,所谓通知线程A,就是唤醒线程A,并让它进入可运行状态。

这种方式的一个好处就是CPU的利用率提高了。

但是也有一些缺点:比如,线程B先执行,一下子添加了5个元素并调用了notify()发送了通知,而此时线程A还执行;当线程A执行并调用wait()时,那它永远就不可能被唤醒了。因为,线程B已经发了通知了,以后不再发通知了。这说明:通知过早,会打乱程序的执行逻辑。

④管道通信

就是使用java.io.PipedInputStream 和 java.io.PipedOutputStream进行通信

具体就不介绍了。分布式系统中说的两种通信机制:共享内存机制和消息通信机制。感觉前面的①中的synchronized关键字和②中的while轮询 “属于” 共享内存机制,由于是轮询的条件使用了volatile关键字修饰时,这就表示它们通过判断这个“共享的条件变量“是否改变了,来实现进程间的交流。

而管道通信,更像消息传递机制,也就是说:通过管道,将一个线程中的消息发送给另一个。

作者:hapjin cnblogs.com/hapjin/p/5492619.html

公众号“Java精选”所发表内容注明来源的,版权归原出处所有(无法查证版权的或者未注明出处的均来自网络,系转载,转载的目的在于传递更多信息,版权属于原作者。如有侵权,请联系,笔者会第一时间删除处理!

最近有很多人问,有没有读者交流群!加入方式很简单,公众号Java精选,回复“加群”,即可入群!

(微信小程序):3000+道面试题,包含Java基础、并发、JVM、线程、MQ系列、Redis、Spring系列、Elasticsearch、Docker、K8s、Flink、Spark、架构设计等,在线随时刷题!

特别推荐:专注分享最前沿的技术与资讯,为弯道超车做好准备及各种开源项目与高效率软件的公众号,「大咖笔记」,专注挖掘好东西,非常值得大家关注。点击下方公众号卡片关注

文章有帮助的话,点在看,转发吧!

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
FIFA最新排名:国足世界第88亚洲第13,阿根廷法国比利时前三

FIFA最新排名:国足世界第88亚洲第13,阿根廷法国比利时前三

直播吧
2024-06-20 15:57:14
仁爱礁菲军舰已加固,甲板上多了起重机和储水塔,短期内不会烂掉

仁爱礁菲军舰已加固,甲板上多了起重机和储水塔,短期内不会烂掉

青年的背包
2024-06-20 21:54:24
乌军创历史 俄遭巨大打击

乌军创历史 俄遭巨大打击

小新在洗手间
2024-06-17 17:17:04
欧洲杯A组积分:瑞士打平即可出线,苏格兰、匈牙利末轮死磕

欧洲杯A组积分:瑞士打平即可出线,苏格兰、匈牙利末轮死磕

直播吧
2024-06-20 05:06:09
杭州一公司拖欠工资并威胁员工?当地街道办:正调查处置

杭州一公司拖欠工资并威胁员工?当地街道办:正调查处置

新京报
2024-06-20 17:22:17
雷军骑单车上班引热议:英国品牌Brmopton!网友:本来还想整个雷总同款呢,一看“小布”卖1.5万

雷军骑单车上班引热议:英国品牌Brmopton!网友:本来还想整个雷总同款呢,一看“小布”卖1.5万

和讯网
2024-06-20 14:49:00
突然确诊癌症,已多处转移!宁波40岁男子悔不当初:膝盖疼了一个月,怎么也没想到会是癌

突然确诊癌症,已多处转移!宁波40岁男子悔不当初:膝盖疼了一个月,怎么也没想到会是癌

鲁中晨报
2024-06-20 15:07:13
为什么那么多人对当今社会不满?最主要就是这7个原因

为什么那么多人对当今社会不满?最主要就是这7个原因

乡野小珥
2024-06-16 12:45:37
潜水时“被弃海中”,浙江籍游客发声:我要的不是赔偿

潜水时“被弃海中”,浙江籍游客发声:我要的不是赔偿

极目新闻
2024-06-20 08:51:50
真被张召忠说中了?掏空国库建造的2艘航母,如今彻底沦为废铁

真被张召忠说中了?掏空国库建造的2艘航母,如今彻底沦为废铁

高山非凡创作
2024-05-16 07:48:51
尴尬,国际乒联向王曼昱发来奥运单打邀请,但女单为孙颖莎、陈梦

尴尬,国际乒联向王曼昱发来奥运单打邀请,但女单为孙颖莎、陈梦

尘语者
2024-06-19 21:45:38
难以相信!日本的国际形象很差吗?网友:佩服得五体投地!

难以相信!日本的国际形象很差吗?网友:佩服得五体投地!

有趣的羊驼
2024-06-06 14:22:52
菲军断指哥:被中国男足迷惑了!一交手兔子变老虎,剑桥大学变脸

菲军断指哥:被中国男足迷惑了!一交手兔子变老虎,剑桥大学变脸

大风文字
2024-06-20 12:38:19
余琦被立案调查!老公身份被扒,前同事曝光真实人品

余琦被立案调查!老公身份被扒,前同事曝光真实人品

吃货的分享
2024-06-20 16:11:31
今天是6月20号晚间,突然曝出1个重大消息,明天要来大动作吗?

今天是6月20号晚间,突然曝出1个重大消息,明天要来大动作吗?

股市皆大事
2024-06-20 15:24:30
羞耻又奇怪的男人生理知识(女生必看)

羞耻又奇怪的男人生理知识(女生必看)

喜马拉雅主播暮霭
2024-06-19 11:54:50
18强完整分档:日伊韩1档,西亚7队3-4档,中国5档,朝鲜印尼6档

18强完整分档:日伊韩1档,西亚7队3-4档,中国5档,朝鲜印尼6档

直播吧
2024-06-20 16:08:13
男性160-190cm标准体重对照表,可能自己并不胖,不用减肥

男性160-190cm标准体重对照表,可能自己并不胖,不用减肥

增肌减脂
2024-06-20 16:28:01
连警方都说了,南海没有斧头帮,只有自拍杆

连警方都说了,南海没有斧头帮,只有自拍杆

三叔的装备空间
2024-06-20 17:55:31
为什么很多足球词汇如今都已不再出现?看完你就明白了

为什么很多足球词汇如今都已不再出现?看完你就明白了

元爸体育
2024-06-15 07:25:02
2024-06-21 02:10:44
Java精选
Java精选
一场永远也演不完的戏
1551文章数 3855关注度
往期回顾 全部

科技要闻

小米SU7流量泼天,富贵却被蔚来接住了

头条要闻

媒体:以为中国会服软 菲在南海主权之争上存低级误判

头条要闻

媒体:以为中国会服软 菲在南海主权之争上存低级误判

体育要闻

千夫所指的关系户 成了拯救葡萄牙的英雄

娱乐要闻

叶舒华参加柯震东生日聚会,五毒俱全

财经要闻

楼市新“王炸”!释放何信号?

汽车要闻

售价11.79-14.39万元 新一代哈弗H6正式上市

态度原创

房产
家居
亲子
手机
旅游

房产要闻

海棠湾!一所重量级国际学校真的来了!

家居要闻

自然开放 实现灵动可变空间

亲子要闻

夫妻俩趁孩子睡着后聊聊一天的琐事,这就是向往的婚姻吧?

手机要闻

真我GT7再次被确认:超声波指纹+新等深超微曲,一加13要有压力了

旅游要闻

铁路儿童票新规 已有超4900万小旅客免费出行

无障碍浏览 进入关怀版