最近看多线程的时候发现对于join的理解有些错误,在网上查了不少资料,根据自己的理解整理了一下,这里之所以把join和wait放在一起,是因为join的底层实现就是基于wait的,一并讲解更容易理解。

wait

了解join就先需要了解wait,wait是线程间通信常用的信号量,作用就是让线程暂时停止运行,等待其他线程使用notify来唤醒或者达到一定条件自己苏醒。

wait是一个本地方法,属于Object类,其底层实现是JVM内部实现,是基于monitor对象监视锁。

//本地方法
public final native void wait(long timeout) throws InterruptedException;
//参数有纳秒和毫秒
public final void wait(long timeout, int nanos) throws InterruptedException {
if (timeout < 0) {
throw new IllegalArgumentException("timeout value is negative");
} if (nanos < 0 || nanos > 999999) {
throw new IllegalArgumentException(
"nanosecond timeout value out of range");
}
//存在纳秒,默认加一毫秒
if (nanos > 0) {
timeout++;
} wait(timeout);
}
//无参数,默认为0
public final void wait() throws InterruptedException {
wait(0);
}

根据源码可以发现,虽然wait有三个重载的方法,但是主要的还是wait(long timeout)这个本地方法,其他两个都是基于这个来封装的,由JVM底层源码不太好看到,我就以流程的形式来描述。

synchronized (this) {
System.out.println("A begin " + System.currentTimeMillis());
try {
wait(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("A end " + System.currentTimeMillis());
}
  • 上述代码在执行到wait(5000)时首先会释放当前占用的锁,并暂停线程。
  • 在暂停的5秒内如果收到其他线程的notify()方法发来的信号,那么就再次尝试获取已经释放的锁
  • 如果获取到那么就继续执行,没有就等待锁释放来竞争。
  • 如果在5秒内未收到信号,那么到时间后就自动苏醒去尝试获取锁。

而对于时间的参数timeout需要注意的是,如果输入0不代表不暂停,而是需要特殊情况自己苏醒或者notify唤醒,这里有个特殊点,wait(0)是可以自己苏醒的

public class Thread2 extends Thread{
private Thread1 a;
public Thread2(Thread1 a) {
this.a = a;
} @Override
public void run() {
synchronized (a) {
System.out.println("B begin " + System.currentTimeMillis());
try {
a.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("B end " + System.currentTimeMillis());
}
}
} public class Thread1 extends Thread{
@Override
public void run() {
synchronized (this) {
System.out.println("A begin " + System.currentTimeMillis());
System.out.println("A end " + System.currentTimeMillis());
}
}
} public class Main{
public static void main(String[] args) {
Thread1 thread = new Thread1();
Thread2 thread2 = new Thread2(thread);
thread2.start();
thread.start();
System.out.println("main end "+System.currentTimeMillis());
}
}

这个例子运行结果存在以下情况

B begin 1494995803564
main end 1494995803565
A begin 1494995803565
A end 1494995803565
B end 1494995803565

wait()在没有notify()情况下自动苏醒了,因此这里可以看到,当前情况下Thread.wait()等待过程中,如果Thread结束了,是可以自动唤醒的。这个会在join中被使用。

join

了解了wait的实现原理之后就可以来看join了,join是Thread类的方法,不是底层本地方法,这里可以看一下它的源码。

public final synchronized void join(long millis)
throws InterruptedException {
long base = System.currentTimeMillis();
long now = 0;
//参数判断<0,抛异常
if (millis < 0) {
throw new IllegalArgumentException("timeout value is negative");
}
//参数为0,即join()
if (millis == 0) {
//当前线程存活,就调用wait(0),一直到调用join的线程结束再自动苏醒
while (isAlive()) {
wait(0);
}
//参数>0,调用wait(long millis)等待一段时间后自动唤醒
} else {
while (isAlive()) {
long delay = millis - now;
if (delay <= 0) {
break;
}
wait(delay);
now = System.currentTimeMillis() - base;
}
}
} public final synchronized void join(long millis, int nanos)
throws InterruptedException { if (millis < 0) {
throw new IllegalArgumentException("timeout value is negative");
} if (nanos < 0 || nanos > 999999) {
throw new IllegalArgumentException(
"nanosecond timeout value out of range");
} if (nanos >= 500000 || (nanos != 0 && millis == 0)) {
millis++;
} join(millis);
} public final void join() throws InterruptedException {
join(0);
}

很明显,join的三个重载方法主要还是基于join(long millis)方法,因此我们主要关注这个方法,方法的处理逻辑如下

  • 判断参数时间参数,如果参数小于0,抛出IllegalArgumentException("timeout value is negative")异常
  • 参数等于0,判断调用join的线程(假设是A)是否存活,不存活就不执行操作,如果存活,就调用wait(0),阻塞join方法,等待A线程执行完在结束join方法。
  • 参数大于0,判断调用join的A线程是否存活,不存活就不执行操作,如果存活,就调用wait(long millis),阻塞join方法,等待时间结束再继续执行join方法。

由于join是synchronized修饰的同步方法,因此会出现join(long millis)阻塞时间超过了millis的值。

public class Thread1 extends Thread{
@Override
public void run() {
synchronized (this) {
System.out.println("A begin " + System.currentTimeMillis());
try {
sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("A end " + System.currentTimeMillis());
}
}
} public class Main{
public static void main(String[] args) {
Thread1 thread = new Thread1();
thread.start();
try {
thread.join(1000);
} catch (InterruptedException e) {
e.printStackTrace();
System.out.println("main end "+System.currentTimeMillis());
}
}

这个例子的运行结果是

A begin 1494996862054
A end 1494996867056
main end 1494996867056

main线程一定是最后执行完的,按照流程来说,1秒之后阻塞就结束了,main线程应该就可以开始执行了,但是这里有一个注意点,join(long millis)在执行millis>0的时候在wait(delay)之后还有一行代码,而上面代码1秒之后只是结束了wait方法,并没有执行完join方法。上面的例子,由于join的锁和thread的锁相同,在thread运行完之前,锁不会释放,那么导致join一直阻塞在最后一步无法结束,才会出现上面的情况。

join和wait的更多相关文章

  1. SQL Server-聚焦IN VS EXISTS VS JOIN性能分析(十九)

    前言 本节我们开始讲讲这一系列性能比较的终极篇IN VS EXISTS VS JOIN的性能分析,前面系列有人一直在说场景不够,这里我们结合查询索引列.非索引列.查询小表.查询大表来综合分析,简短的内 ...

  2. SQL Server-聚焦NOT IN VS NOT EXISTS VS LEFT JOIN...IS NULL性能分析(十八)

    前言 本节我们来综合比较NOT IN VS NOT EXISTS VS LEFT JOIN...IS NULL的性能,简短的内容,深入的理解,Always to review the basics. ...

  3. Nested Loops join时显示no join predicate原因分析以及解决办法

    本文出处:http://www.cnblogs.com/wy123/p/6238844.html 最近遇到一个存储过程在某些特殊的情况下,效率极其低效, 至于底下到什么程度我现在都没有一个确切的数据, ...

  4. c# Enumerable中Aggregate和Join的使用

    参考页面: http://www.yuanjiaocheng.net/ASPNET-CORE/asp.net-core-environment.html http://www.yuanjiaochen ...

  5. 超详细mysql left join,right join,inner join用法分析

    下面是例子分析表A记录如下: aID        aNum 1           a20050111 2           a20050112 3           a20050113 4   ...

  6. join Linq

    List<Publisher> Publishers = new List<Publisher>(); Publisher publish1 = new Publisher() ...

  7. mysql join 和left join 对于索引的问题

    今天遇到一个left join优化的问题,搞了一下午,中间查了不少资料,对MySQL的查询计划还有查询优化有了更进一步的了解,做一个简单的记录: select c.* from hotel_info_ ...

  8. BCL中String.Join的实现

    在开发中,有时候会遇到需要把一个List对象中的某个字段用一个分隔符拼成一个字符串的情况.比如在SQL语句的in条件中,我们通常需要把List<int>这样的对象转换为“1,2,3”这样的 ...

  9. [数据库基础]——图解JOIN

    阅读导航 一.概要 二.JOIN分类 三.JOIN分类详解 一.概要 JOIN对于接触过数据库的人,这个词都不陌生,而且很多人很清楚各种JOIN,还有很多人对这个理解也不是很透彻,这次就说说JOIN操 ...

  10. Spark join 源码跟读记录

    PairRDDFunctions类提供了以下两个join接口,只提供一个参数,不指定分区函数时默认使用HashPartitioner;提供numPartitions参数时,其内部的分区函数是HashP ...

随机推荐

  1. 隐马尔科夫模型HMM(一)HMM模型

    隐马尔科夫模型HMM(一)HMM模型基础 隐马尔科夫模型HMM(二)前向后向算法评估观察序列概率 隐马尔科夫模型HMM(三)鲍姆-韦尔奇算法求解HMM参数(TODO) 隐马尔科夫模型HMM(四)维特比 ...

  2. 每天一个JS 小demo之韩雪冬轮播图。主要知识点:html,css布局,对于数组和对象的理解和运用

    @charset "utf-8"; /* CSS Document */ ;; } li { list-style: none; } img { border: none; } b ...

  3. Java通过jxl解析Excel文件入库,及日期格式处理方式 (附源代码)

    JAVA可以利用jxl简单快速的读取文件的内容,但是由于版本限制,只能读取97-03  xls格式的Excel. 本文是项目中用到的一个实例,先通过上传xls文件(包含日期),再通过jxl进行读取上传 ...

  4. WCF入门的了解准备工作

    了解WCF, 及WCF入门需要掌握哪里基本概念? 1.准备工作 >1.1 . XML >1.2 . Web Service >1.3 . 远程处理 (RPC) >1.4.  消 ...

  5. AngularJS高级程序设计读书笔记 -- 过滤器篇

    一. 过滤器基础 过滤器用于在视图中格式化展现给用户的数据. 一旦定义过滤器之后, 就可在整个模块中全面应用, 也就意味着可以用来保证跨多个控制器和视图之间的数据展示的一致性. 过滤器将数据在被指令处 ...

  6. SICP-1.7-递归函数

    递归函数 函数内部直接或间接的调用函数自身 将复杂问题简单化 例子程序 def sum_digits(n): """Return the sum of the digit ...

  7. ASP.NET Core 源码学习之 Options[2]:IOptions

    在上一篇中,介绍了一下Options的注册,而使用时只需要注入IOption即可: public ValuesController(IOptions<MyOptions> options) ...

  8. 推荐几款.NET客户端开源报表图

    如果你正在开发客户端报表图相关的应用,除了.NET自带的控件,你还可以考虑使用以下几个控件库. [OxyPlot] OxyPlot是一个支持.NET的跨平台绘图库.你可以在很多平台上使用它,如WPF, ...

  9. indexOf和lastIndexOf的使用

    indexOf()和 lastIndexOf()是返回位置index的两个方法:都是接收两个参数,其中,indexOf()方法从数组的开头(位 置 0)开始向后查找:lastIndexOf()方法则从 ...

  10. 新安装mysql 第三方工具连接不上问题

    Mysql从客户端连接服务器连不上的问题   公司要用Mysql做一个测试,开始在自己的本地建一个Mysql数据库自己本地的程序再连上去,没有遇到过连接不上的问题.这次数据库在服务器上,从本地客户端连 ...