多线程中的wait与join

本来是想把标题定为《Thread的wait与join》,后来想想不严谨,因为wait是Object的方法,不是Thread独有的,所以这里要注意一下。
关于wait()方法,在Object中有三个重载方法:

    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);
    }

    public final void wait() throws InterruptedException {
        wait(0);
    }

无论是哪个,最终都是要调用本地方法 wait(long timeout) 。
开局一张图:


image.png

今天主要就来讲一下wait(),join()的关系吧。

wait

一个Object的方法,目的是将调用obj.wait()的线程置为waiting的状态,等待其他线程调用obj.notify()或者obj.notifyAll()来唤醒。最常见的就是生产者/消费者功能。

有一点注意的就是,wait/notify方法的调用必须处在该对象的锁(Monitor)中,也即,在调用这些方法时首先需要获得该对象的锁。否则会抛出IllegalMonitorStateException异常。

wait/notify的通俗解释:
1.线程A首先获取到obj的锁,然后执行了obj.wait(),这个方法就会是线程A暂时让出对obj锁的持有,并把线程A转换waiting状态,同时加入锁对象的等待队列。
2.线程B获取到了obj的锁,然后执行了obj.notify(),这个方法通知了锁对象的等待队列,使正在等待队列中的线程A改为阻塞状态,使A进入对obj锁的竞争。当然在执行notify后并不会使线程A马上获取到锁,因为线程B目前还在持有obj的锁。
3.线程A获取到obj锁,继续从wait()之后的代码运行。

举个例子:

public class ThreadTest {

    static final Object obj = new Object();  //对象锁

    private static boolean flag = false;

    public static void main(String[] args) throws Exception {

        Thread consume = new Thread(new Consume(), "Consume");
        Thread produce = new Thread(new Produce(), "Produce");
        consume.start();
        Thread.sleep(1000);
        produce.start();

        try {
            produce.join();
            consume.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    // 生产者线程
    static class Produce implements Runnable {

        @Override
        public void run() {

            synchronized (obj) {
                System.out.println("进入生产者线程");
                System.out.println("生产");
                try {
                    TimeUnit.MILLISECONDS.sleep(2000);  //模拟生产过程
                    flag = true;
                    obj.notify();  //通知消费者
                    TimeUnit.MILLISECONDS.sleep(1000);  //模拟其他耗时操作
                    System.out.println("退出生产者线程");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    //消费者线程
    static class Consume implements Runnable {

        @Override
        public void run() {
            synchronized (obj) {
                System.out.println("进入消费者线程");
                System.out.println("wait flag 1:" + flag);
                while (!flag) {  //判断条件是否满足,若不满足则等待
                    try {
                        System.out.println("还没生产,进入等待");
                        obj.wait();
                        System.out.println("结束等待");
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                System.out.println("wait flag 2:" + flag);
                System.out.println("消费");
                System.out.println("退出消费者线程");
            }

        }
    }
}

结果:

进入消费者线程
wait flag 1:false
还没生产,进入等待
进入生产者线程
生产 // 1秒后notify
退出生产者线程 //生产线程结束
结束等待
wait flag 2:true
消费
退出消费者线程

可以看到,生产线程在notify后,消费线程并没有马上继续运行,原因就是上面提到的第二点。

wait()方法有了一个大概的了解,下面看看join的原理。

join

join方法通常的解释就是等待调用线程执行完毕后,再继续执行当前线程。通常用在多线程的业务上,某个线程的运算需要另一个线程的结果时,就可以使用join。

来个例子:

public class JoinMainDemo {
    public static void main(String[] args) throws InterruptedException {
        Thread thread1 = new Thread(new Runnable() {
            public void run() {
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("thread 1");
            }
        });
        Thread thread2 = new Thread(new Runnable() {
            public void run() {
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("thread 2");
            }
        });
        Thread thread3 = new Thread(new Runnable() {
            public void run() {
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("thread 3");
            }
        });

        thread1.start();
        thread1.join();
        thread2.start();
        thread2.join();
        thread3.start();
        thread3.join();

        System.out.println("thread Main");
    }
}

情况1输出:

thread 1
thread 2
thread 3
thread Main

流程:Main线程调用thread1join后,会等待join执行完毕才会继续运行,thread2,thread3都是这样。1,2,3顺序是固定的。

我们改一下代码顺序

        thread1.start();
        thread3.start();
        thread2.start();
        thread1.join();
        thread2.join();
        thread3.join();

情况2输出:

thread 3
thread 1
thread 2
thread Main

结果是线程1,2,3随机顺序,Main一定在最后。

去掉join:

        thread1.start();
        thread3.start();
        thread2.start();

情况3输出:

thread Main
thread 3
thread 1
thread 2

这个就是普通的线程执行结果。

我们来分析一下每种情况的原因。join的行为像不像被wait后自动释放的过程?看一下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;
            }
        }
    }

可见,线程A在调用obj.join方法时,obj线程如果是alive状态(线程开启start,但未结束),那么就执行wait方法。同时看join(long m)方法是synchronized修饰的,这是我们使用wait时需要先获取锁的前置条件。既然知道了join的内核是wait方法,通过对wait的了解,线程A此时是waiting的状态,并进入了obj锁的等待队列排队去。那么是谁在什么时候释放了线程A呢?

这时要了解一下Thread.exit()方法:

/**
* 这个方法由系统调用,当该线程完全退出前给它一个机会去释放空间。
*/
 private void exit() {
        if (group != null) {
            group.threadTerminated(this);
            group = null;
        }
        /* Aggressively null out all reference fields: see bug 4006245 */
        target = null;
        /* Speed the release of some of these resources */
        threadLocals = null;
        inheritableThreadLocals = null;
        inheritedAccessControlContext = null;
        blocker = null;
        uncaughtExceptionHandler = null;
    }

这个方法由系统调用,当该线程完全退出前给它一个机会去释放空间。再往下跟到threadTerminated(this)方法

    void threadTerminated(Thread t) {
        synchronized (this) {
            remove(t);

            if (nthreads == 0) {
                notifyAll();
            }
            if (daemon && (nthreads == 0) &&
                (nUnstartedThreads == 0) && (ngroups == 0))
            {
                destroy();
            }
        }
    }

这里有一个notifyAll()的方法,可见就是在这里是线程A得到了释放继续运行。

到这里就解释通了,join的确是通过wait方法使调用线程变为等待状态,再在被调用线程运行结束时通过系统调用exit方法启动了notifyAll。

我们可以这么理解,把obj.join方法替换成obj.wait()方法,并且在obj线程运行结束后自动执行notifyAll()方法,这样就可以用wait的思路来理解join的运行过程了。

好了,现在回到上面例子的三种情况:
情况1: 可以发现,thread2.start前有thread1.join,thread3.start前有thread2.join,这样就能解释thread1,2,3是按顺序执行的。thread1.join的时候,Main线程进入了thread1锁对象的等待队列,只有thread1运行完成后才会得到释放。进而才会按顺序开启线程,thread2.start,thread3.start。

情况2:先将3个线程开启,再依次执行join。join之前,三个线程已经都在运行,所以输出的顺序并没有固定,只是会控制Main线程运行时间。Main的输出肯定是在最后的。当然我们可以把thread2的sleep时间调大一点,并且不执行join再来运行看看结果,自己尝试解释一下。

情况3:没有join,各个线程独自运行,互不影响。

总结

1.wait的注意点: wait方法是Object的方法; wait/notify方法需要获得对象锁后执行。
2.wait方法会把调用线程转为等待waiting状态,释放对象锁,并进入对象锁的等待队列。
3.notify/notifyAll方法会唤醒对象锁的等待队列,使其中的线程进入阻塞blocking状态抢占对象锁。
4.调用notify/notifyAll后,在notify/notifyAll的前获得的对象锁得到释放后,等待队列里的线程才有机会抢占锁继续执行。
5.join方法的内核就是wait。在被调用对象的线程运行完毕后,系统自动调 用被调用对象notifyAll方法。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容