前面介绍了多线程并发之时的资源抢占情况,以及利用同步、加锁、信号量等机制解决资源冲突问题,不过这些机制只适合同一资源的共享分配,并未涉及到某件事由的前因后果。日常生活中,经常存在两个前后关联的事务,像雇员和雇主这两个角色,他们之间的某些工作就带有因果关系。比如要等雇主接到了项目,雇员才有活干;又如每月末员工都等着老板发工资,这样才有钱逛街和吃大餐,此时员工的消费行为便依赖于老板的发薪水动作。如此看来,两个线程之间理应建立某种消息通路,每当线程A完成某个事项,就将完成标志通知线程B,线程B收到通知之后,认为前提条件已经满足,这才进行后续的处理过程。线程之间的消息通路,可视作在线程间传递信息,专业的说法叫做“通信”,如何在多线程并发时进行有效通信,这是多线程技术中的一大课题。
依据线程并发时的不同管理机制,线程间的通信也各有不同的方式,接下来将分别论述同步机制与加锁机制之下的两种线程通信过程。
首先是同步机制,采用同步代码块的话,需要在关键字synchronized后面补充待同步的对象实例,之前的同步代码块统一写成“synchronized (this)”。可是圆括号内部一定要填this吗?圆括号的内部参数究竟是干什么用的?其实synchronized附带的圆括号参数正是在线程间通信的邮差,以前的同步演示代码由于没进行线程通信,因此圆括号里的参数没有具体要求,一般填this即可。现在要想在线程间进行通信,就必须启用圆括号参数了,并且两个线程都要在synchronized后面填写该参数对象。
举个例子,雇员等着雇主发工资,那员工怎样才知道老板已经发了呢?要是由员工自己一会儿一会儿去查银行卡,平时的工作都会受到影响,所以可让员工留个等工资的心眼就好。然后老板一个一个发工资,发完之后给员工递个工资条,或者给员工发封工资邮件,这样员工收到工资条便知薪水到账了。那么在等工资和发工资这两个线程之间,即可令工资条作为二者的信使,于是同步代码块可改写为“synchronized (工资条对象)”的形式。同时工资条对象还要支持等待与发放两个动作,因为这类动作早就隐藏在Object类的基本方法中,所以开发者不必担心工资条对象该为Integer类型还是别的什么类型,凡是正常的实例都拥有等待与发放的方法,具体的方法说明如下:
wait:等待通知。
notify:在等待队列中随机挑选一个线程发放通知。
notifyAll:向等待队列中的所有线程发放通知。
在编码实现同步机制的通信过程时,先分别创建雇员和雇主的工作任务,其中雇员任务在同步代码块中调用工资条对象的wait方法,表示等着发工资;而雇主任务在同步代码块中调用工资条对象的notify方法,表示发完工资了。然后依次启动员工线程和老板线程,员工线程负责等工资以及收到工资后的消费行为,老板线程负责发工资以及记账操作。据此编写的同步线程通信代码示例如下:

	// 员工与老板之间通过工资条通信
private static Integer salary = 5000; // 测试通过wait和notify方法进行线程间通信
private static void testWaitNotify() {
// 创建雇员的工作任务
Runnable employee = new Runnable() {
@Override
public void run() {
PrintUtils.print(Thread.currentThread().getName(), "等着发工资。");
synchronized (salary) { // 工资是我的,你们别抢
try {
salary.wait(); // 等待发工资
// 打印拿到工资后的庆祝日志
PrintUtils.print(Thread.currentThread().getName(), "今晚赶紧吃大餐。");
} catch (InterruptedException e) { // 等待期间允许接收中断信号
e.printStackTrace();
}
}
}
};
// 创建雇主的工作任务
Runnable boss = new Runnable() {
@Override
public void run() {
// 稍等一会儿,老板线程的同步代码块务必在员工线程的同步代码块之后开始运行,否则员工线程将一直等待
wait_a_moment();
PrintUtils.print(Thread.currentThread().getName(), "开始发工资。");
synchronized (salary) { // 由我发工资,你们别闹
wait_a_moment(); // 银行转账也需要时间
salary.notify(); // 随机通知其中一个等待线程
// 手好酸,发工资也是个体力活,记个账
PrintUtils.print(Thread.currentThread().getName(), "发完工资了。");
}
}
};
new Thread(employee, "同步机制的员工").start(); // 启动员工等工资的线程
new Thread(boss, "同步机制的老板").start(); // 启动老板发工资的线程
} // 稍等一会儿,模拟日常事务的时间消耗
private static void wait_a_moment() {
int delay = new Random().nextInt(500); // 生成500以内的随机整数
try {
Thread.sleep(delay); // 睡眠若干毫秒
} catch (InterruptedException e) {
}
}

运行上面的线程通信代码,打印出以下的线程日志:

14:37:29.685 同步机制的员工 等着发工资。
14:37:29.994 同步机制的老板 开始发工资。
14:37:30.120 同步机制的老板 发完工资了。
14:37:30.120 同步机制的员工 今晚赶紧吃大餐。

从日志可见,员工线程果然在等到工资之后才去吃大餐。

同步机制能够通过wait/notify完成线程通信功能,那么加锁机制又该如何进行线程间通信呢?既然加锁机制设计了专门的锁工具,那么锁钥内外的线程也只能通过锁工具来通信,信使则为调用锁对象的newCondition方法返回的Condition条件对象。条件对象同样拥有等待与发放的方法,且与Object类的三个方法一一对应,具体说明如下:
await:等待通知。
signal:在等待队列中随机挑选一个线程发放通知。
signalAll:向等待队列中的所有线程发放通知。
以可重入锁ReentrantLock为例,依然要先分别创建雇员和雇主的工作任务,其中雇员任务在加锁之后再调用条件对象的await方法,表示等着发工资;而雇主任务在加锁之后再调用条件对象的signal方法,表示发完工资了;另外雇员任务和雇主任务均需在结束之前进行解锁。然后依次启动员工线程和老板线程,员工线程负责等工资以及收到工资后的消费行为,老板线程负责发工资以及记账操作。下面是在加解锁线程之间进行通信的代码例子:

	// 创建一个可重入锁
private final static ReentrantLock reentrantLock = new ReentrantLock();
// 获取可重入锁的条件对象
private static Condition condition = reentrantLock.newCondition(); // 测试通过Condition对象进行线程间通信
private static void testCondition() {
// 创建雇员的工作任务
Runnable employee = new Runnable() {
@Override
public void run() {
PrintUtils.print(Thread.currentThread().getName(), "等着发工资。");
reentrantLock.lock(); // 对可重入锁加锁
try {
condition.await(); // 这里在等待条件对象的信号
// 打印拿到工资后的庆祝日志
PrintUtils.print(Thread.currentThread().getName(), "今晚赶紧吃大餐。");
} catch (InterruptedException e) { // 等待期间允许接收中断信号
e.printStackTrace();
}
reentrantLock.unlock(); // 对可重入锁解锁
}
};
// 创建雇主的工作任务
Runnable boss = new Runnable() {
@Override
public void run() {
// 稍等一会儿,老板线程的加锁务必在员工线程的加锁之后执行,否则员工线程将一直等待
wait_a_moment();
PrintUtils.print(Thread.currentThread().getName(), "开始发工资。");
reentrantLock.lock(); // 对可重入锁加锁
wait_a_moment(); // 银行转账也需要时间
condition.signal(); // 给条件对象发送信号
// 手好酸,发工资也是个体力活,记个账
PrintUtils.print(Thread.currentThread().getName(), "发完工资了。");
reentrantLock.unlock(); // 对可重入锁解锁
}
};
new Thread(employee, "加锁机制的员工").start(); // 启动员工等工资的线程
new Thread(boss, "加锁机制的老板").start(); // 启动老板发工资的线程
}

运行上述的线程通信代码,打印出如下的线程日志:

14:57:07.794 加锁机制的员工 等着发工资。
14:57:07.801 加锁机制的老板 开始发工资。
14:57:07.905 加锁机制的老板 发完工资了。
14:57:07.906 加锁机制的员工 今晚赶紧吃大餐。

可见加锁机制同样实现了线程间通信的功能。

更多Java技术文章参见《Java开发笔记(序)章节目录

Java开发笔记(一百零三)线程间的通信方式的更多相关文章

  1. Java开发笔记(九十三)深入理解字节缓存

    前面介绍了文件通道的读写操作,其中用到字节缓存ByteBuffer,它是位于通道内部的存储空间,也是通道唯一可用的存储形式.ByteBuffer有两种构建方式,一种是调用静态方法wrap,根据输入的字 ...

  2. Java开发笔记(一百零四)普通线程池的运用

    前面介绍了线程的基本用法,以及多线程并发的问题处理,但实际开发中往往存在许多性质相似的任务,比如批量发送消息.批量下载文件.批量进行交易等等.这些同类任务的处理流程一致,不存在资源共享问题,相互之间也 ...

  3. Java开发笔记(一百零五)几种定时器线程池

    前面介绍了普通线程池的用法,就大多数任务而言,它们对具体的执行时机并无特殊要求,最多是希望早点跑完早点出结果.不过对于需要定时执行的任务来说,它们要求在特定的时间点运行,并且往往不止运行一次,还要周期 ...

  4. Java开发笔记(一百零六)Fork+Join框架实现分而治之

    前面依次介绍了普通线程池和定时器线程池的用法,这两种线程池有个共同点,就是线程池的内部线程之间并无什么关联,然而某些情况下的各线程间存在着前因后果关系.譬如人口普查工作,大家都知道我国总人口为14亿左 ...

  5. Java开发笔记(一百零二)信号量的请求与释放

    前面介绍了同步与加锁两种并发处理机制,虽然加锁比起同步要灵活一些,但是加锁在某些高级场合依然力有未逮,包括但不限于下列几点:1.某块代码被加锁之后,对其它线程而言就处于繁忙状态,缺乏弹性的阈值范围:2 ...

  6. Java开发笔记(一百零一)通过加解锁避免资源冲突

    前面介绍了如何通过线程同步来避免多线程并发的资源冲突问题,然而添加synchronized的方式只在简单场合够用,在一些高级场合就暴露出它的局限性,包括但不限于下列几点:1.synchronized必 ...

  7. Java开发笔记(一百)线程同步synchronized

    多个线程一起办事固然能够加快处理速度,但是也带来一个问题:两个线程同时争抢某个资源时该怎么办?看来资源共享的另一面便是资源冲突,正所谓鱼与熊掌不可兼得,系统岂能让多线程这项技术专占好处?果然是有利必有 ...

  8. Java开发笔记(一百零九)XML报文的定义和解析

    前面介绍了JSON格式的报文解析,虽然json串短小精悍,也能有效表达层次结构,但是每个元素只能找到对应的元素值,不能体现更丰富的样式特征.比如某个元素除了要传输它的字符串文本,还想传输该文本的类型. ...

  9. Java开发笔记(一百三十)Swing的选择框

    不管是AWT还是Swing,都把选择框分成两类:复选框和单选按钮,这两类控件无论是外观上还是功能上均有显著差异.例如,在外观方面,复选框是在方框内打勾,而单选按钮是在圆圈内画圆点:在功能方面,复选框允 ...

随机推荐

  1. qt4.8.5 qtwebkit 静态编译 版本

    2013年就编译好了,qtwebkit是最不好编译的了,尤其是静态编译,这儿分享给大家 估计总有人会用得到... 静态库下载地址:http://yunpan.cn/cyyNqrApbVDwq  提取码 ...

  2. Python中str、list、numpy分片操作

    在Python里,像字符串(str).列表(list).元组(tupple)和这类序列类型都支持切片操作 对对象切片,s是一个字符串,可以通过类似数组索引的方式获取字符串中的字符,同时也可以用s[a: ...

  3. laravel5.2总结--数据填充

      1 生成一个seeder文件 你可以通过 make:seeder artisan命令来生成一个 Seeder.所有通过框架生成的 Seeder 都将被放置在 database/seeds 路径: ...

  4. SSH进阶之路

    [SSH进阶之路]Hibernate基本原理(一)       在开始学Hibernate之前,一直就有人说:Hibernate并不难,无非是对JDBC进一步封装.一句不难,难道是真的不难还是眼高手低 ...

  5. 56、使用android studio(v1.3.*)修改包名 (rename package name)

    一.修改包名 ①选中目录,开始构造 在弹窗中选中Rename directory 在弹窗中选中Rename package 填写新的包名,点击Refactor 如果有警告,不用管它,直接点击Do Re ...

  6. 48、android代码架构总结

    之前是按功能模块进行分类,现在随着功能模块越来越多,代码层次不再清晰,所以修改了工程结构: 之前: 经过修改现在: 1.更严谨的遵循mvc架构 bean目录存放的是数据模型 ui存储的是activit ...

  7. Solr 配置连接数据库

    前面我们将solr安装并创建了core同时也配置可IK分词器,接下来我们通过配置连接Mysql数据库并把数据导入到solr(使用ik分词器). 1.配置managed-schema文件 Request ...

  8. 求解Catalan数,(大数相乘,大数相除,大数相加)

    Catalan数 卡塔兰数是组合数学中一个常在各种计数问题中出现的数列.以比利时的数学家欧仁·查理·卡塔兰(1814–1894)命名.历史上,清代数学家明安图(1692年-1763年)在其<割圜 ...

  9. POJ 3368:Frequent values(线段树区间合并)

    题目大意,给出一段非降序列,求一些区间中出现频率最高的数的出现次数. 分析: 显然,区间中一个数多次出现必然是连续的,也就是最长的连续相等的一段. 用线段树解决,维护三个信息:一个区间最长连续的区间的 ...

  10. arcgis engine10.1和arcObjects的一些问题

    1.arcengine10.1只支持vs2010 2.10.1以后没有engine runtimes,改成engine了,以前的engine可以理解为Arcobject,就是我们可以只装AO