线程间的协作(1)——等待/通知机制

简介: 参考资料《Java并发编程的艺术》《Java编程思想》《Java核心技术》

当使用线程来同时运行多个任务时,可以使用锁(互斥)来同步两个任务的行为,从而使得一个任务不会干扰另一个任务,也就是说两个任务在交替的使用某个共享资源,使用互斥可以保证时刻只有一个任务可以访问这项资源。

    解决了多线程引起的资源竞争问题,那么我们又该如何实现线程间的彼此协作?使得多个任务可以一起去解决某个问题,也就是说,现在的问题不是线程间的干涉,而是线程间如何协调,因为在这类问题中,某些部分必须在其他部分被解决之前解决。

    一个线程修改了一个对象的值,而另一个线程感知到了变化,然后进行相应的操作,整个过程开始于一个线程,而最终执行又是另一个线程。前者是生产者,后者就是消费者,这种模式隔离了“做什么”(what)和“怎么做”(How),在功能层面上实现了解耦,体系结构上具备了良好的伸缩性,在Java语言中实现的简单的办法是让消费者线程不断地循环检查变量是否符合预期,如下面代码所示,在 while循环中设置不满足的条件,如果条件满足则退出while循环,从而完成消费者的工作。

while (value != desire) {
   Thread.sleep(1000);
}
doSomething();

    但是上述代码也会带来其他问题

  • 难以确保及时性。在睡眠时,基本不消耗处理器资源,但是如果睡得过久,就不能及时发现条件已经变化,也就是及时性难以保证。
  • 难以降低开销。如果降低睡眠的时间,比如休眠1毫秒,这样消费者能更加迅速地发现条件变化,但是却可能消耗更多的处理器资源,造成了无端的浪费。

    所以,我们需要Java内置的等待/通知机制来更好的解决这个问题。实现等待/通知机制的相关方法是所以对象都具有的,因为他们被定义在了Object对象上。

1.概念

    等待/通知机制,是指一个线程A调用了对象O的wait()方法进入等待状态,而另一个线程B 调用了对象O的notify()或者notifyAll()方法,线程A收到通知后从对象O的wait()方法返回,进而 执行后续操作。上述两个线程通过对象O来完成交互,而对象上的wait()和notify/notifyAll()的 关系就如同开关信号一样,用来完成等待方和通知方之间的交互工作。(也可以通过Condition类的await(),signal()/signalAll()来实现)

2. 相关API 

    1.wait()

    当对象调用该方法的线程会无限期进入等待状态(或者说在等待着返回,也就是return,类似于网络编程中的ServerSocket对象的accept方法,只有一个客户端建立连接时方法才能结束,更直白的说相当于在wait()方法内部有一个while死循环,循环跳出条件就是这个对象被其他线程唤醒,解释代码如下),并释放对象的锁,只有等待另一个线程的通知或者被中断该方法才会返回,注意!返回之后并不是一定会进入运行状态,而是先变成就绪状态,然后再继续运行。

//当然,内部实现并不是这样,仅仅作为一个比喻
private boolean token=false;
//线程A调用对象的wait1方法,陷入等待
public void wait1(){
	while(token==false){
		
	}
	return;
}
//线程B调用同一个对象的notify1方法,便可实现唤醒
public void notify1(){
	token=true;
}

    2.wait(long time),wait(long time,int  nanos)

    超时等待,等待另一个线程的通知或在等待一段时间后自动方法返回,time是毫秒,nanos是纳秒。

    3.notify()

    通知一个在该对象上等待的线程,使其从wait()方法返回,而前提是当前线程获取了同一对象的锁。

    4.notifyAll()

    通知所有在该对象上等待的线程,使其从wait()方法返回。

3.详解

    1)为何wait()和notify/notifyAll()方法时放在Object类中而不是Thread类中,这三种方法都是针对线程的功能却作为所有类的通用功能来实现,因为等待和唤醒必须是同一个锁。而锁可以是任意对象,所以可以被任意对象调用的方法是定义在object类中。既然wait()和notify/notifyAll()必须操作锁,那么wait()和notify/notifyAll()必须处于同步代码块或方法中。(Lock锁中使用Condition来实现)

public class WaitNotify {
	static boolean flag = true;
	static Object lock = new Object();

	public static void main(String[] args) throws Exception {
		Thread waitThread = new Thread(new Wait(), "WaitThread");
		waitThread.start();
		TimeUnit.SECONDS.sleep(1);
		Thread notifyThread = new Thread(new Notify(), "NotifyThread");
		notifyThread.start();
	}

	static class Wait implements Runnable {
		public void run() {
			// 加锁,拥有lock的Monitor
			synchronized (lock) {
				// 当条件不满足时,继续wait,同时释放了lock的锁
				while (flag) {
					try {
						System.out.println(Thread.currentThread() + " flag is true. wait"
								+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
						lock.wait();
					} catch (InterruptedException e) {
					}
				}
				// 条件满足时,完成工作
				System.out.println(Thread.currentThread() + " flag is false. running"
						+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
			}
		}
	}

	static class Notify implements Runnable {
		public void run() {
			// 加锁,拥有lock的Monitor
			synchronized (lock) {
				// 获取lock的锁,然后进行通知,通知时不会释放lock的锁,
				// 直到当前线程释放了lock后,WaitThread才能从wait方法中返回
				System.out.println(Thread.currentThread() + " hold lock. notify @ "
						+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
				lock.notifyAll();
				flag = false;

			}
			// 再次加锁
			synchronized (lock) {
				System.out.println(Thread.currentThread() + " hold lock again. sleep"
						+ new SimpleDateFormat("HH:mm:ss").format(new Date()));

			}
		}
	}
}

    2)使用wait()/notify()时的细节:

  • 使用wait()、notify()和notifyAll()时需要先对调用对象加锁。
  • 调用wait()方法后,线程状态由RUNNING变为WAITING,并将当前线程放置到对象的等待队列。
  • notify()或notifyAll()方法调用后,等待线程不会立即从wait()返回,需要调用notify()或 notifAll()的线程释放锁之后,等待线程才有机会从wait()返回。
  • notify()方法将等待队列中的一个等待线程从等待队列中移到同步队列中,而notifyAll() 方法则是将等待队列中所有的线程全部移到同步队列,被移动的线程状态由WAITING变为 BLOCKED。
  • 从wait()方法返回的前提是获得了调用对象的锁。

    3)等待/通知机制的经典范式

        等待方遵循如下原则:

  • 获取对象的锁。
  • 如果条件不满足,那么调用对象的wait()方法,被通知后仍要检查条件。
  • 条件满足则执行对应的逻辑。

        伪代码为:

synchronized(对象) {
    while(条件不满足) {
        对象.wait();
    }
    对应的处理逻辑
}

        通知方应遵循:

  • 获得对象的锁。
  • 改变条件。
  • 通知所有等待在对象上的线程。

        伪代码为:

synchronized(对象) {
    改变条件
    对象.notifyAll();
}

4. 第二种实现方法Lock与Condition

    除了使用Object类中的wait()/notify()外,还可以通过Lock与Condition结合使用实现。具体使用与wait()/notify()相似。

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class AwaitAndSignal {
	static boolean flag = true;
	static Lock lock = new ReentrantLock();
	static Condition con=lock.newCondition();
	public static void main(String[] args) throws Exception {
		Thread AwaitThread = new Thread(new Await(), "AwaitThread");
		AwaitThread.start();
		TimeUnit.SECONDS.sleep(1);
		Thread SignalThread = new Thread(new Signal(), "SignalThread");
		SignalThread.start();
	}

	static class Await implements Runnable {
		public void run() {
			try{
				lock.lock();
				while(flag&&!Thread.interrupted()){
					try {
						System.out.println(Thread.currentThread() + " flag is true. wait"
								+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
						con.await();
						
					} catch (InterruptedException e) {
						// TODO Auto-generated catch block
						e.printStackTrace();
					}
				}
				System.out.println(Thread.currentThread() + " flag is false. running"
						+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
			}finally {
				lock.unlock();
			}
		}
	}

	static class Signal implements Runnable {
		public void run() {
			try{
				lock.lock();
				System.out.println(Thread.currentThread() + " hold lock. notify"
						+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
				con.signalAll();
				flag = false;
			}finally {
				lock.unlock();
			}
			// 再次加锁
			try{
				lock.lock();
				System.out.println(Thread.currentThread() + " hold lock again. sleep"
						+ new SimpleDateFormat("HH:mm:ss").format(new Date()));
			}finally {
				lock.unlock();
			}
		}
	}
}

5.Thread.join()方法

    如果一个线程A执行了thread.join()语句,其含义是:当前线程A等待thread线程终止之后才从thread.join()返回。线程Thread除了提供join()方法之外,还提供了join(long millis)和join(longmillis,int nanos)两个具备超时特性的方法。

public class ThreadMethodJoin {
	public static void main(String[] args) throws Exception {
		Thread previous = Thread.currentThread();
		for (int i = 0; i < 10; i++) {
		// 每个线程拥有前一个线程的引用,需要等待前一个线程终止,才能从等待中返回
		Thread thread = new Thread(new Domino(previous), String.valueOf(i));
		thread.start();
		previous = thread;
		}
		TimeUnit.SECONDS.sleep(5);
		System.out.println(Thread.currentThread().getName() + " terminate.");
	}
	static class Domino implements Runnable {
		private Thread thread;
		public Domino(Thread thread) {
			this.thread = thread;
		}
		public void run() {
			try {
				thread.join();
			} catch (InterruptedException e) {
			}
			System.out.println(Thread.currentThread().getName() + " terminate.");
		}
	}

}

 

相关文章
|
6月前
|
Java 开发者
多线程编程范式(一) 协作范式
多线程编程范式(一) 协作范式
|
安全 Java
【JavaSE专栏76】三态和五态,线程的不同状态:新建、运行、状态、阻塞、等待、计时等待状态
【JavaSE专栏76】三态和五态,线程的不同状态:新建、运行、状态、阻塞、等待、计时等待状态
104 0
|
6月前
|
数据处理
多线程与并发编程【线程对象锁、死锁及解决方案、线程并发协作、生产者与消费者模式】(四)-全面详解(学习总结---从入门到深化)
多线程与并发编程【线程对象锁、死锁及解决方案、线程并发协作、生产者与消费者模式】(四)-全面详解(学习总结---从入门到深化)
67 1
|
3月前
|
Java 开发者
解锁并发编程新姿势!深度揭秘AQS独占锁&ReentrantLock重入锁奥秘,Condition条件变量让你玩转线程协作,秒变并发大神!
【8月更文挑战第4天】AQS是Java并发编程的核心框架,为锁和同步器提供基础结构。ReentrantLock基于AQS实现可重入互斥锁,比`synchronized`更灵活,支持可中断锁获取及超时控制。通过维护计数器实现锁的重入性。Condition接口允许ReentrantLock创建多个条件变量,支持细粒度线程协作,超越了传统`wait`/`notify`机制,助力开发者构建高效可靠的并发应用。
90 0
|
4月前
|
安全 Java API
Java并发编程的艺术:解锁多线程同步与协作的秘密
【7月更文挑战第28天】在Java的世界中,并发编程如同一场精心编排的交响乐,每一个线程都是乐团中的乐手,而同步机制则是那指挥棒,确保旋律的和谐与统一。本文将深入探讨Java并发编程的核心概念,包括线程的创建、同步机制、以及线程间的通信方式,旨在帮助读者解锁Java多线程编程的秘密,提升程序的性能和响应性。
43 3
|
4月前
|
安全 Java 数据处理
Java并发编程:线程同步与协作的深度解析
在探索Java并发编程的海洋中,线程同步与协作的灯塔指引着航向。本文将深入挖掘线程同步机制的核心原理,揭示锁、条件变量等工具如何确保数据的一致性和线程间有序的通信。通过案例分析,我们将解码高效并发模式背后的设计哲学,并探讨现代Java并发库如何简化复杂的同步任务。跟随文章的步伐,您将获得提升多线程应用性能与可靠性的关键技能。 【7月更文挑战第24天】
45 5
|
4月前
|
Java C# Python
线程等待(Thread Sleep)
线程等待(Thread Sleep)
|
4月前
|
测试技术
三种等待方式(‌线程等待、‌隐式等待、‌显式等待)
三种等待方式(‌线程等待、‌隐式等待、‌显式等待)
214 4
|
5月前
|
安全 Java
使用notifyAll唤醒所有等待线程
使用notifyAll唤醒所有等待线程
|
4月前
|
安全 Java
使用notifyAll唤醒所有等待线程
使用notifyAll唤醒所有等待线程