3

这里的代码

class Aux implements Runnable {

    private Boolean isOn = false;
    private String statusMessage;
    private final Object lock;

    public Aux(String message, Object lock) {
        this.lock = lock;
        this.statusMessage = message;
    }

    @Override
    public void run() {
        for (;;) {
            synchronized (lock) {

                if (isOn && "left".equals(this.statusMessage)) {
                    isOn = false;
                    System.out.println(statusMessage);
                } else if (!isOn && "right".equals(this.statusMessage)) {
                    isOn = true;
                    System.out.println(statusMessage);
                }

                if ("left".equals(this.statusMessage)) {
                    System.out.println("left " + isOn);
                }

            }
        }
    }
}

public class Question {
    public static void main(String [] args) {
        Object lock = new Object();

        new Thread(new Aux("left", lock)).start();
        new Thread(new Aux("right", lock)).start();
    }
}

在这段代码中,我希望看到:

左,右,左,右等等,

但是当带有“左”消息的线程将 isOn 更改为 false 时,带有“右”消息的线程看不到它并且我得到(“右真”和“左假”控制台消息),左线程不会在真中获得 isOn ,但正确的线程无法更改它,因为它总是看到旧的 isOn 值(true)。

当我将 volatile 修饰符添加到 isOn 时,没有任何变化,但是如果我将 isOn 更改为带有布尔字段的某个类并更改此字段,那么线程会看到更改并且它工作正常

提前致谢。

4

1 回答 1

7

isOn是一个线程局部变量,所以当一个线程修改它时,另一个线程将看不到更改。当您使用带有布尔字段的类时,您可能会为每个线程提供该类的相同实例,这意味着线程将看到其他线程所做的更改。但是,您创建了两个实例Aux而不是只使用一个。

如果您希望它在两个不同的实例上工作,请尝试将static修饰符(连同volatile修饰符)添加到该Boolean字段;否则,不要创建第二个Aux.

于 2013-06-29T22:24:45.590 回答