线程间的通讯
2019-04-22 / JAVA / 1187 次围观 / 0 次吐槽 /什么时候需要通信
多个线程并发执行时, 在默认情况下CPU是随机切换线程的,如果我们希望他们有规律的执行, 就可以使用通信, 例如每个线程执行一次打印
线程怎么通信
如果希望线程等待, 就调用wait()
如果希望唤醒等待的线程, 就调用notify();
notify是随机唤醒一个线程
notifyAll是唤醒所有线程
这两个方法必须在同步代码中执行, 并且使用同步锁对象来调用
如果方法中没有同步锁,会有异常IllegalMonitorStateException
案例:两个线程间的通讯
public class Demo {
public static void main(String[] args) {
//1.创建任务对象
MyTask task = new MyTask();
//2.开启两个线程执行2个任务
new Thread(){
public void run() {
while(true){
try {
task.task1();
} catch (InterruptedException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
};
}.start();
new Thread(){
public void run() {
while(true){
try {
task.task2();
} catch (InterruptedException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
};
}.start();
}
}
class MyTask{
//标识 1:可以执行任务1,2:可以执行任务2
int flag = 1;
public synchronized void task1() throws InterruptedException{
if(flag != 1){
this.wait();//当前线程等待
}
System.out.println("任务1");
flag = 2;
this.notify();//唤醒其它线程
}
public synchronized void task2() throws InterruptedException{
if(flag != 2){
this.wait();//线程等待
}
System.out.println("任务2");
flag = 1;
this.notify();//唤醒其它线程
}
}线程通讯的一些疑问
1.在同步代码块中,用哪个对象锁,就用哪个对象调用wait方法
2.为什么wait方法和notify方法定义在Object这类中?
因为锁对象可以是任意对象,Object是所有的类的基类,所以wait方法和notify方法需要定义在Object这个类中
3.sleep方法和wait方法的区别?
sleep方法必须传入参数,参数就是时间,时间到了自动醒来
wait方法可以传入参数也可以不传入参数,传入参数就是在参数的时间结束后等待,不传入参数就是直接等待
sleep方法在同步函数或同步代码块中,不释放锁,睡着了也抱着锁睡
wait方法在同步函数或者同步代码块中,释放锁
JDK1.5新特性互斥锁
ReentrantLock介绍
使用ReentrantLock类也可以实现同步加锁
ReentrantLock叫[互斥锁],使用lock()和unlock()方法进行同步
使用ReentrantLock类使用要点
使用ReentrantLock类的newCondition()方法可以获取Condition对象
需要等待的时候使用Condition的await()方法, 唤醒的时候用signal()方法
不同的线程使用不同的Condition, 这样就能区分唤醒的时候找哪个线程了
案例:
/**
* 互斥锁的使用步骤
* 1.创建互斥锁对象
* 2.创建3个Condition
* 3.加锁、解锁
* 4.线程等待-Condition的await方法
* 5.线程唤醒-Condition的signal方法
* @author gyf
*
*/
class MyTask{
//创建互斥锁对象
ReentrantLock rl = new ReentrantLock();
//创建3个Condition
Condition c1 = rl.newCondition();
Condition c2 = rl.newCondition();
Condition c3 = rl.newCondition();
//标识 1:可以执行任务1,2:可以执行任务2, 3:可以执行任务3
int flag = 1;
public void task1() throws InterruptedException{
rl.lock();//加锁
if(flag != 1){
c1.await();//当前线程等待
}
System.out.println("任务1");
flag = 2;
//指定唤醒线程2
c2.signal();
rl.unlock();//解锁
}
public void task2() throws InterruptedException{
rl.lock();
if(flag != 2){
c2.await();//线程等待
}
System.out.println("任务2");
flag = 3;
//唤醒线程3
c3.signal();
rl.unlock();
}
public void task3() throws InterruptedException{
rl.lock();
if(flag != 3){
c3.await();//线程等待
}
System.out.println("任务3");
flag = 1;
//唤醒线程1
c1.signal();
rl.unlock();
}
}Powered By Cheug's Blog
Copyright Cheug Rights Reserved.