Java中Thread的join方法为什么能让线程插队?
编辑:业余草
来源:https://www.xttblog.com/?p=5062
Java中Thread的join方法为什么能让线程插队?
这个问题很多高级工程师可能都不会,因为平时很少用到。
但是在面试中就有可能经常会遇到这样的问题:在主线程中有两个子线程,如何能让着两个子线程能顺序的执行?答案自然是用 join 来使得两个线程顺序执行
。说到这里,我前面也有类似的文章《让线程按顺序执行 8 种方法》,最近又有粉丝问到,所以还是推荐大家看一看。
今天这个问题,我们再来通过代码来搞懂它!
public class JoinThread {
public static void main(String[] args) throws Exception {
Thread codedq = new MyThread("业余草");
Thread xttblog = new MyThread("公众号");
codedq.start();
codedq.join();
xttblog.start();
}
@Data
static class MyThread extends Thread {
private String userName;
public MyThread(String userName) {
this.userName = userName;
}
@Override
public void run() {
try {
for (int i = 0; i < 5; i++) {
System.out.println(userName + " - " + i);
Thread.sleep(1000);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
当每个线程启动后,分别打印五次信息,通过不同的名字来区分是哪个线程打印的。最终的执行结果如下:
业余草 - 0
业余草 - 1
业余草 - 2
业余草 - 3
业余草 - 4
公众号 - 0
公众号 - 1
公众号 - 2
公众号 - 3
公众号 - 4
通过最终的结果可以看到 join 可以使得两个线程是顺序执行,那为什么 join 能控制线程顺序执行呢,我们看下 join 的具体实现!
//外部调用的方法
public final void join() throws InterruptedException {
join(0);
}
//内部的具体实现
public final synchronized void join(long millis) throws InterruptedException {
long base = System.currentTimeMillis();
long now = 0;
if (millis < 0) {
throw new IllegalArgumentException("timeout value is negative");
}
if (millis == 0) {
while (isAlive()) {
wait(0);
}
} else {
while (isAlive()) {
long delay = millis - now;
if (delay <= 0) {
break;
}
wait(delay);
now = System.currentTimeMillis() - base;
}
}
}
通过源码可知,join 通过 synchronized 关键字来保证线程安全,主线程在调用了 codedq.start() 之后调用了 codedq.join(),当 codedq 线程未执行完成时,主线程会被以下代码阻塞。
if (millis == 0) {//join()方法默认milis为0
while (isAlive()) {//线程未执行完成,此条件为true
wait(0);//等待notify
}
}
当 codedq 线程执行完成之后,此线程的生命周期即将结束,在生命周期结束前,codedq 线程会使用 notifyAll() 方法,通知所有正在等待该对象锁的线程(我即将死去,你们不要再等了)。wait(0) 接收到 notify 之后,会再次进行 isAlive() 判断,codedq 死亡之后,就跳出循环,join 方法结束,之后就继续执行主线程中的其他代码。
同时我们也能看到 join 方法里面能传递时间参数,大概作用就是等待指定时间之后,如果之前线程还未执行完成,那么久不再等待。
综上所述,这个面试题并不难。主要难在两点,一是 join 这个方法,平时不常用;二是,就算我们用过 join,但却很少有人去剖析它的源码,思考它的底层实现。
这篇文章有粉丝在面试中遇到,刚好今天周末,撸一篇文章大家共勉,以后再有人遇到此类问题,我就可以把这篇文章甩给他了。如果你也有卡壳的问题,不妨私信我,在空闲时间里为你排忧解难!
- zookeeper curator处理会话过期session expired
- redis事务
- 数据库表反向生成(一) MyBatis-generator与IDEA的集成
- 数据库表反向生成(二) Django ORM inspectdb
- RabbitMQ与AMQP协议
- 大数据算法设计模式(2) - 左外链接(leftOuterJoin) spark实现
- hs_err_pid
- django celery的分布式异步之路(二) 高并发
- django celery的分布式异步之路(一) 起步
- SpringMVC拦截器Interceptor
- 元宵快乐:看SQL大师们用SQL绘制的团圆
- Python Redis pipeline操作
- python concurrent.futures
- Deepmind的星际争霸2强化学习教程(1):建立环境与训练模型
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- 基于DOM4J的XML文件解析类
- Win7安装和配置Apache2.4服务器的详细方法
- shiro会话管理示例代码
- Windows Apache2.4 VC9(ApacheHaus)详细安装配置教程
- 在centos 7中安装配置k8s集群的步骤详解
- Centos7.2 编译安装方式搭建 phpMyAdmin
- CentOS 6.5 web服务器apache的安装与基本设置
- Linux本机与服务器文件互传及Linux服务器文件上传下载命令写法
- linux利用read命令获取变量中的值
- 解决Centos7 安装腾达U12无线网卡驱动问题
- CentOS 6.5上编译安装Apache服务器的方法(最小化安装)
- 固定QPS压测模式探索
- Centos6 网络配置的实例详解
- centos6.5升级安装配置supervisor的教程
- Linux的路由表详细介绍