多线程基础问题待解决处理

为什么出现有线程没有结束,
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循环,但是不对

我想要达到的结果:"不能设置while (num <= 3)而使用while (true)的原因是"

有问题的代码在这里,两个线程轮流执行并且交替被唤醒,最先结束的线程,拿你那个3来说,1,2,1的执行顺序,当1第二次执行完以后唤醒2,此时number>3,while直接就出来了, 导致线程1一直在等待别人唤醒,

img


所以你将else的代码,单独拿出来,并且再调用一次唤醒方法,那就行了

img

我的理解:
1、使用while (num <= 3)的话,由于while内的if条件也是(num <= 3),所以只要能进入循环if条件就一定成立,会执行if条件成立的代码,即每次循环结束都会有一个线程会处于等待状态,当 num = 4 时,while循环条件不成立,退出循环,但此时还有一个线程是处于等待状态并没有被唤醒的,所以运行结果是打印3行且程序未停止。
2、使用while (true)的话,只要循环内部不break或者return,则会一直循环。只要进入循环线程就会被唤醒,所以当 num = 4 时,还会进入循环唤醒线程,但此时if条件不成立,走else的break退出循环。此时将没有等待状态的线程,程序运行结束。