您的位置:首页 > 编程语言 > Java开发

java中关于线程间协作所用关键字synchronized,wait,notify的用法

2016-06-30 17:50 615 查看
wait/notify()关键字适用于一个线程通知另一个线程所需的条件状态已就绪,最常用于线程在循环中休眠直到获取特定条件的场景. 例如,一个线程一直等待直到队列中有一个组件能够处理;当组件添加到队列时,另一个线程能够唤醒这个等待的线程。

如下代码演示线程间的协作:

package thread_waitnotify;

public class ProductFactory {
class Worker implements Runnable {
private final Object lockObject = new Object();
private volatile boolean hasWork = false;

private void waitTakeTask() {
synchronized (lockObject) {
while (!hasWork) {
try {
System.out.println("等待中...");
lockObject.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
takeTask();
}
}

private void takeTask() {
boolean tmpHasWork = false;
synchronized (lockObject) {
if (hasWork)
tmpHasWork = true;
}
if (tmpHasWork) {
System.out.println("开始做任务...");
try {
Thread.sleep(3000);
synchronized (lockObject) {
hasWork = false;
}
} catch (InterruptedException e) {
e.printStackTrace();
}

System.out.println("任务完成,休息3s后退出...");
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}

// System.out.println("任务完成,继续寻找下一项任务...");
// takeTask();
} else {
System.out.println("暂时没有任务,进入等待状态...");
waitTakeTask();
}
}

public void notifyChange() {
synchronized (lockObject) {
hasWork = true;
System.out.println("起床开工了...");
lockObject.notifyAll();// 唤醒所有等待获取lockObject锁的线程
}
}

@Override
public void run() {
takeTask();
}
}

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

Worker worker = new ProductFactory().new Worker();
Thread thread = new Thread(worker);
thread.start();
Thread.sleep(3000);// 主线程休眠3s后唤醒子线程
worker.notifyChange();// 主线程设置子线程中的信号量并唤醒阻塞的线程

thread.join();// 主线程阻塞,等待子线程执行完成后继续主线程的执行
System.out.println("子线程执行完成,主线程退出。");
System.exit(0);
}

}


执行结果如下所示:



说一下synchronized和wait()、notify()的关系:

1.有synchronized的地方不一定有wait,notify

2.有wait,notify的地方必有synchronized.这是因为wait和notify不是属于线程类,而是每一个对象都具有的方法,而且,这两个方法都和对象锁有关,有锁的地方,必有synchronized。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: