三线程打印ABC的面试题

正确的代码是这样的:
public class MutliThreadPrintABC implements Runnable{

//打印内容
String content;
//两个锁
Object prev;
Object self;

public MutliThreadPrintABC(String content,Object prev,Object self) {
    this.content=content;
    this.prev=prev;
    this.self=self;
}

@Override
public void run() {
    for(int i=0;i<10;i++){
        synchronized (prev) {
            System.out.println(Thread.currentThread().getName()+"获取到了前一个线程的锁");
            synchronized (self) {
                System.out.println(content);
                self.notify();
            }
            try {
                prev.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

public static void main(String[] args) throws InterruptedException {
    Object a = new Object();
    Object b = new Object();
    Object c = new Object();
    Thread A = new Thread(new MutliThreadPrintABC("A",c,a));
    A.setName("A");
    Thread B = new Thread(new MutliThreadPrintABC("B",a,b));
    B.setName("B");
    Thread C = new Thread(new MutliThreadPrintABC("C",b,c));
    C.setName("C");
    A.start();
    Thread.sleep(100);
    B.start();
    Thread.sleep(100);
    C.start();
}

}
我在学习这个的时候写了个错误的版本就是释放锁这一部分prev.wait()写到了最里面的同步块里:
public void run() {
for(int i=0;i<10;i++){
synchronized (prev) {
System.out.println(Thread.currentThread().getName()+"获取到了前一个线程的锁");
synchronized (self) {
System.out.println(content);
self.notify();
try {
prev.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
这个时候程序运行会卡住,而且是A线程执行完就直接执行了C线程,而且都只打印了一次就卡住了,我实在搞不懂为什么B线程没拿到A线程执行完释放的锁,这个错误版本到底程序是怎么执行的,本人小白没干过多线程的工作,正在学习,求大佬解释一下

prev.wait();写在synchronized (prev)里面
一旦prev被锁,它就没有机会再执行出来了。

看我博客吧 这个用ReentrantLock和Condition来实现。

http://blog.csdn.net/m0_37741173/article/details/79319755?from=singlemessage&isappinstalled=0