在多线程情况下,如果一个线程进入了if语句块中,在if语句块中执行了wait方法,该线程处于等待(wait)状态,如果该线程被唤醒(notify),不管if的条件是否发生了变化,该线程会执行wait之后的代码,并且会跳出if语句块执行if语句块外面的代码
如果是while,被唤醒了之后,会继续执行wait方法之后的代码,代码块里面的代码执行完了会继续判断条件,如果为true,会继续执行while语句块里面的代码,如果为false,才会执行判断语句块后面的代码
下面是演示代码
package unit_3.while_if_difference;
/**
* @author xiaogang
* @date 2019/2/21 10:10
*/
public class ValueObject {
public static boolean b = true;
}
package unit_3.while_if_difference;
/**
* @author xiaogang
* @date 2019/2/20 18:47
*/
public class ThreadA extends Thread {
private Object lock;
public ThreadA(Object b) {
this.lock = b;
}
@Override
public void run() {
System.out.println("ThreadA.run");
synchronized (lock) {
try {
System.out.println("if before");
if (ValueObject.b) {
System.out.println("wait before");
lock.wait();
System.out.println("wait end");
}
System.out.println("if after");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
package unit_3.while_if_difference;
/**
* @author xiaogang
* @date 2019/2/21 10:11
*/
public class ThreadB extends Thread{
private Object lock;
public ThreadB(Object lock) {
this.lock = lock;
}
@Override
public void run() {
synchronized (lock) {
System.out.println("唤醒一个lock的一个等待状态的线程 start");
lock.notify();
System.out.println("唤醒一个lock的一个等待状态的线程 end");
}
}
}
package unit_3.while_if_difference;
/**
* @author xiaogang
* @date 2019/2/20 18:46
*/
public class Run {
public static void main(String[] args) throws InterruptedException {
String lock = new String("");
ThreadA threadA = new ThreadA(lock);
threadA.start();
ThreadB threadB = new ThreadB(lock);
Thread.sleep(10);
ValueObject.b = true;
threadB.start();
}
}
执行结果是:
将ThreadA中if修改为while
package unit_3.while_if_difference;
/**
* @author xiaogang
* @date 2019/2/20 18:47
*/
public class ThreadA extends Thread {
private Object lock;
public ThreadA(Object b) {
this.lock = b;
}
@Override
public void run() {
System.out.println("ThreadA.run");
synchronized (lock) {
try {
System.out.println("if before");
while (ValueObject.b) {
System.out.println("wait before");
lock.wait();
System.out.println("wait end");
}
System.out.println("if after");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
结果如下:
这才是正确的结果