Java多线程-线程的交互

转载 2013年12月02日 15:56:02

一、线程交互的基础知识
void notify():唤醒在此对象监视器上等待的单个线程。 
void notifyAll():唤醒在此对象监视器上等待的所有线程。 
void wait():导致当前的线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法。

当然,wait()还有另外两个重载方法:
void wait(long timeout):导致当前的线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法,或者超过指定的时间量。 
void wait(long timeout, int nanos):导致当前的线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法,或者其他某个线程中断当前线程,或者已超过某个实际时间量。

关于等待/通知,要记住的关键点是:
必须从同步代码块内调用wait()、notify()、notifyAll()方法。线程不能调用对象上等待或通知的方法,除非它拥有那个对象的锁。

wait()、notify()、notifyAll()都是Object的实例方法。与每个对象具有锁一样,每个对象可以有一个线程列表,他们等待来自该信号(通知)。线程通过执行对象上的wait()方法获得这个等待列表。从那时候起,它不再执行任何其他指令,直到调用对象的notify()方法为止。如果多个线程在同一个对象上等待,则将只选择一个线程(不保证以何种顺序)继续执行。如果没有线程等待,则不采取任何特殊操作。

例子

package thread;

/**
 * 计算1+2+3 ... +100的和
 * 
 * @author administrator
 * 
 */
public class ThreadSum extends Thread {
	int total = 0;

	@Override
	public void run() {
		synchronized (this) {
			for (int i = 0; i < 101; i++) {
				 total += i;
			}
			 //(完成计算了)唤醒在此对象监视器上等待的单个线程,在本例中线程ThreadInteractionTest被唤醒
			notify();
		}
	}
}

package thread;

/**
 * 线程的交互
 * 
 * @author administrator
 * 
 */
public class ThreadInteractionTest {
	public static void main(String[] args) {
		ThreadSum sum = new ThreadSum();
		// 启动计算线程
		sum.start();
		// 线程ThreadInteractionTest拥有sum对象上的锁。
		// 线程为了调用wait()或notify()方法,该线程ThreadInteractionTest必须是那个对象锁的拥有者
		synchronized (sum) {

			try {
				System.out.println("等待对象sum完成计算。。。");
				// 当前线程ThreadInteractionTest等待
				sum.wait();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}

			System.out.println("sum对象计算的总和是:" + sum.total);
		}
	}

}

运算结果:

等待对象sum完成计算。。。
sum对象计算的总和是:5050

注意:当在对象上调用wait()方法时,执行该代码的线程立即放弃它在对象上的锁。然而调用notify()时,并不意味着这时线程会放弃其锁。如果线程仍然在完成同步代码,则线程在移出之前不会放弃锁。因此,只要调用notify()并不意味着这时该锁变得可用。

二、多个线程在等待一个对象锁时使用notifyAll()
在多数情况下,最好通知等待某个对象的所有线程。如果这样做,可以在对象上使用notifyAll()让所有在此对象上等待的线程冲出等待区,返回到可运行状态。

package thread;

/**
 * 计算1+2+3 ... +100的和
 * 
 * @author administrator
 * 
 */
public class ThreadSum2 extends Thread {
	int total = 0;

	@Override
	public void run() {
		synchronized (this) {
			for (int i = 0; i < 101; i++) {
				total += i;
			}
			// 通知所有在此对象上等待的线程
			notifyAll();
		}
	}
}

package thread;

public class ThreadInteractionTest2 extends Thread {
	ThreadSum2 sum;

	public ThreadInteractionTest2(ThreadSum2 sum) {
		this.sum = sum;
	}

	@Override
	public void run() {
		synchronized (sum) {
			try {
				System.out.println("等待对象sum完成计算。。。");
				// 当前线程ThreadInteractionTest等待
				sum.wait();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("sum对象计算的总和是:" + sum.total);
		}
	}

	public static void main(String[] args) {
		ThreadSum2 sum = new ThreadSum2();

		// 启动三个线程,分别获取计算结果
		new ThreadInteractionTest2(sum).start();
		new ThreadInteractionTest2(sum).start();
		new ThreadInteractionTest2(sum).start();

		// 启动计算线程
		sum.start();

	}
}

运行结果:


等待对象sum完成计算。。。
等待对象sum完成计算。。。
等待对象sum完成计算。。。
sum对象计算的总和是:5050
sum对象计算的总和是:5050
sum对象计算的总和是:5050
谈一下synchronized和wait()、notify()等的关系:
1、有synchronized的地方不一定有wait,notify
2、有wait,notify的地方必有synchronized.这是因为wait和notify不是属于线程类,而是每一个对象都具有的方法,而且,这两个方法都和对象锁有关,有锁的地方,必有synchronized。
另外,注意一点:如果要把notify和wait方法放在一起用的话,必须先调用notify后调用wait,因为如果调用完wait,该线程就已经不是currentthread了。

java线程通信,解决线程之间的交互问题

线程通信: 注意三个都是Object的方法 并且都必须在synchronzied代码块和安全方法下使用否则会报异常 wiat:使当前线程挂起,释放锁,其他线程可以参与进来共享其数据。 notif...
  • Liutt55
  • Liutt55
  • 2015年11月29日 21:40
  • 1290

java多线程:线程的交互

一,线程交互的基础知识         在java.lang.Object包中
  • carl_jiang
  • carl_jiang
  • 2014年08月28日 09:16
  • 1084

java多线程学习(四)——线程的交互

线程交互中用到的三个基本函数: void notify();唤醒在此对象监视器上等待的单个线程。 void notifyAll();唤醒在此对象监视器上等待的所有线程。 void wait();...
  • u013599970
  • u013599970
  • 2014年09月21日 01:17
  • 2043

Java同步和线程交互的理解

Java的线程设计比较简单,可是从操作系统层面的API来理解是,会存在问题。Java提供的线程设计和Unix/Linux系统的API设计思路稍有不同,在理解上也会存在问题。笔者使用Java多年,经常使...
  • yin138
  • yin138
  • 2017年03月20日 01:03
  • 363

Java线程(六):线程的交互

一、线程交互的基础知识线程交互知识点需要从java.lang.Object的类的三个方法来学习:void notify() 唤醒在此对象监视器上等待的单个线程。 void no...
  • gusgao
  • gusgao
  • 2016年10月27日 13:21
  • 202

java多线程同步以及线程间通信详解&消费者生产者模式&死锁&Thread.join()(多线程编程之二)

多线程系列教程: java多线程-概念&创建启动&中断&守护线程&优先级&线程状态(一) java多线程同步以及线程间通信详解&消费者生产者模式&死锁&Thread.join()(二) 本篇我...
  • javazejian
  • javazejian
  • 2016年03月13日 16:58
  • 10275

java swt多线程问题的解决现场(一点一点耐心看)

一点一点耐心看,重要的是解决问题的过程
  • liuxiaogangqq
  • liuxiaogangqq
  • 2015年11月23日 19:01
  • 1628

多线程的GUI—Java

多线程实现的计数的秒表控制程序。package GUIThread; import java.awt.Container; import java.awt.FlowLayout; import ja...
  • qq_30507287
  • qq_30507287
  • 2016年10月30日 19:50
  • 755

【Android笔记】多线程与 UI 交互的方法

一共四种: 1.Activity.runOnUIThread(Runnable action) 2.View.post(Runnable action),View.postDelay(Runnable...
  • kinglearnjava
  • kinglearnjava
  • 2015年09月16日 20:32
  • 1078

SWT/Jface UI线程和Java Thread的交互问题

我们在操作UI时,难免要调用一些非UI的方法来获取数据 我们还要将数据返回到对应的UI控件上,又不能出现那种假死-一直等待的状态。 解决的办法: 我们操作时应该启动2个线程, 1个负责后台去获...
  • soszou
  • soszou
  • 2012年09月25日 15:18
  • 2337
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java多线程-线程的交互
举报原因:
原因补充:

(最多只允许输入30个字)