package javase10;
public class Thread10 {
public static void main(String[] args) {
ThreadCommunication tc1 = new ThreadCommunication();
Thread t1 = new Thread(tc1);
t1.setName("线程1");
Thread t2 = new Thread(tc1);
t2.setName("线程2");
t1.start();
t2.start();
}
}
class ThreadCommunication implements Runnable {
private int num = 1;
public Object obj = new Object();
@Override
public void run() {
while (num <= 3) {
synchronized (obj) {
obj.notify();//唤醒被wait()的一个线程,如果是线程1唤醒线程2,但此时同步监视器被线程1占用,因此线程1继续往下执行
if (num <= 3) {
try {
Thread.sleep(10);//增加异常概率,因此要加同步锁
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + ",即将要操作数" + num);
num++;
try {
obj.wait();//使得调用如下wait()方法的线程进入阻塞状态,线程1执行完后进入阻塞状态并且释放同步监视器,如此循环便是两个线程轮流执行1~100的打印
} catch (InterruptedException e) {
e.printStackTrace();
}
} else {
break;//终止循环
}
}
}
}
}
线程2,即将要操作数1
线程1,即将要操作数2
线程2,即将要操作数3
假如最后一个打印结果是:"线程2,即将要操作数100",意味着i的值随着i++变成了101,线程2后面执行了wait()进入阻塞阶段并释放同步监控器,线程1获取同步监控器,然后释放线程2的阻塞状态,线程2判断while (num <= 100)不符合后结束了while循环,线程1判断num不符合if (num <= 100)执行了break中止了该线程的while循环,但是不对
有问题的代码在这里,两个线程轮流执行并且交替被唤醒,最先结束的线程,拿你那个3来说,1,2,1的执行顺序,当1第二次执行完以后唤醒2,此时number>3,while直接就出来了, 导致线程1一直在等待别人唤醒,
我的理解:
1、使用while (num <= 3)的话,由于while内的if条件也是(num <= 3),所以只要能进入循环if条件就一定成立,会执行if条件成立的代码,即每次循环结束都会有一个线程会处于等待状态,当 num = 4 时,while循环条件不成立,退出循环,但此时还有一个线程是处于等待状态并没有被唤醒的,所以运行结果是打印3行且程序未停止。
2、使用while (true)的话,只要循环内部不break或者return,则会一直循环。只要进入循环线程就会被唤醒,所以当 num = 4 时,还会进入循环唤醒线程,但此时if条件不成立,走else的break退出循环。此时将没有等待状态的线程,程序运行结束。