Java多线程-线程的交互

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

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

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

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

例子线程

package cn.thread;

/**
 * 计算1+2+3 ... +100的和
 * 
 * @author 林计钦
 * @version 1.0 2013-7-23 上午10:06:04
 */
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 cn.thread;

/**
 * 线程的交互
 * 
 * @author 林计钦
 * @version 1.0 2013-7-23 上午10:04:11
 */
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()并不意味着这时该锁变得可用。

2、多个线程在等待一个对象锁时使用notifyAll()
在多数状况下,最好通知等待某个对象的全部线程。若是这样作,能够在对象上使用notifyAll()让全部在此对象上等待的线程冲出等待区,返回到可运行状态。code

package cn.thread;

/**
 * 计算1+2+3 ... +100的和
 * 
 * @author 林计钦
 * @version 1.0 2013-7-23 上午10:06:04
 */
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 cn.thread;


/**
 * 线程的交互
 * 
 * @author 林计钦
 * @version 1.0 2013-7-23 上午10:04:11
 */
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()等的关系:
一、有synchronized的地方不必定有wait,notify
二、有wait,notify的地方必有synchronized.这是由于wait和notify不是属于线程类,而是每个对象都具备的方法,并且,这两个方法都和对象锁有关,有锁的地方,必有synchronized。
另外,注意一点:若是要把notify和wait方法放在一块儿用的话,必须先调用notify后调用wait,由于若是调用完wait,该线程就已经不是currentthread了。对象

相关文章
相关标签/搜索