多线程通讯的三大法器,你真的会用吗?

wait, notify, notifyAll 是多线程之间通讯最重要的 3 个方法,今天,栈长给你们普及一下它们的知识要点及应用实战。java

定义

wait:让持有该对象锁的线程等待;多线程

notify: 唤醒任何一个持有该对象锁的线程;线程

notify: 唤醒全部持有该对象锁的线程;code

它们 3 个的关系是,调用对象的 wait 方法使线程暂停运行,经过 notify/ notifyAll 方法唤醒调用 wait 暂时的线程。对象

然而,它们并非 Thread 类中的方法,而是 Object 类中的,为何呢!? 由于每一个对象都有监视锁,线程要操做某个对象固然是要获取某个对象的锁了,而不是线程的锁。blog

如图所示,wait 带时间表示最大超时时间,过了时间还不唤醒就会自动唤醒线程从新竞争对象锁。资源

几个重要的点

一、调用对象的 wait, notify, notifyAll 方法须要拥有对象的监视器锁,即它们只能在同步方法(块)中使用;get

二、调用 wait 方法会使用线程暂停并让出 CPU 资源,同时释放持有的对象的锁;同步

三、多线程使用 notify 容易发生死锁,通常使用 notifyAll;it

四、关于 wait 和 sleep 的详细区别请翻阅 《多线程 sleep 和 wait 的 5 个区别》这篇文章。

实战

/**
 * 公众号:Java技术栈
 */
public static void main(String[] args) {
	Object lock = new Object();
	Thread t1 = new Thread(() -> {
		synchronized (lock) {
			for (int i = 0; i < 20; i++) {
				System.out.print(i);
				if (i == 10) {
					try {
						lock.wait();
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			}
		}
	});

	Thread t2 = new Thread(() -> {
		synchronized (lock) {
			try {
				Thread.sleep(5000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.print("Java技术栈");
			lock.notifyAll();
		}
	});

	t1.start();
	t2.start();
}

上面的例子结合 wait/ notifyAll 来演示了它们的相互做用。

线程 t1 首先输出 012345678910,5秒后继续输出 Java技术栈111213141516171819


更多 Java 多线程技术文章请在Java技术栈公众号后台回复关键字:多线程。

本文原创首发于公众号:Java技术栈(id:javastack),关注公众号在后台回复 "多线程" 可获取更多,转载请原样保留本信息。

相关文章
相关标签/搜索