问题形容:

启动两个线程,

  • 线程A打印数组{"A","B","C","D","E","F","G"};
  • 线程B打印数组{"1","2","3","4","5","6","7"};

要求,打印进去的后果是

A 1 B 2 C 3 D 4 E 5 F 6 G 7
剖析
用到的知识点:synchronized、notify、wait

代码

public class Demo {    private static String array1[] = {"A","B","C","D","E","F","G"};    private static String array2[] = {"1","2","3","4","5","6","7"};    public static void main(String[] args) {        //定义同一把锁        Object o = new Object();        //线程1        new Thread(() -> {            for (int i = 0; i < array1.length; i++) {                synchronized (o) {                    System.out.print(array1[i] + " ");                    o.notify();                    try {                        o.wait();                    } catch (InterruptedException e) {                        e.printStackTrace();                    }                }            }        }).start();        //线程2        new Thread(() -> {            for (int i = 0; i < array2.length; i++) {                synchronized (o) {                    System.out.print(array2[i] + " ");                    o.notify();                    try {                        o.wait();                    } catch (InterruptedException e) {                        e.printStackTrace();                    }                }            }        }).start();    }}

知识点总结

wait notify notifyAll

// 都是被 `final` 润饰的办法,不能被重写。public final native void notify();public final void wait() throws InterruptedException {    wait(0);}public final native void notifyAll();
  • 当线程执行 wait() 办法时,会开释以后的锁,而后让出CPU,进入期待状态。
  • 只有当 notify/notifyAll 被执行的时候,才会唤醒一个或多个正在处于期待状态的线程,而后持续往下执行,直到执行完synchronized代码块的代码或是中途遇到wait(),再次开释锁。
  • notify() 办法的执行只是唤醒沉睡的线程,而不会立刻开释锁。

notify() 和 notifyAll() 的区别

  • notify 办法只唤醒一个期待线程并使该线程开始执行。
  • notifyAll 办法会唤醒所有期待线程

经典案例
生产者-消费者问题

Java多线程学习之wait、notify/notifyAll 详解