java - 当标志在不同线程中更改时循环不会结束

标签 java multithreading while-loop

<分区>

我在 Java 程序的 main 方法中运行了一个 while 循环。该循环应该一直运行,直到在程序的 keyPressed 方法中将 boolean 标志变量设置为 true(我将程序作为 KeyListener 添加到 JFrame)。

import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import javax.swing.JFrame;

public class ThreadWhile implements KeyListener {
    private boolean flag = false;

    public static void main(String[] args) {
        //Set up key listening on a dummy JFrame
        JFrame frame = new JFrame("Key Detector 9000");
        frame.setVisible(true);
        ThreadWhile tw = new ThreadWhile();
        frame.addKeyListener(tw);

        System.out.println("Looping until flag becomes true...");
        while(!tw.flag) {
            //Commenting the println out makes the loop run forever!
            System.out.println(tw.flag); 
        }
        System.out.println("Flag is true, so loop terminated.");
        System.exit(0);
    }

    public void keyPressed(KeyEvent e) {
        flag = true;
        System.out.println("flag: " + flag);
    }

    public void keyReleased(KeyEvent e) {}
    public void keyTyped(KeyEvent e) {}
}

我的理解是 keyPressed 方法在它们自己的线程中执行,所以看起来当我按下一个键时,变量“flag”应该设置为 true,并且在 main 方法中运行的 while 循环应该结束。

但是,当我运行这个程序时,循环永远运行,即使我们可以看到“标志”变量被正确设置为真!奇怪的是,如果我在 while 循环中插入“flag”变量的快速 System.out.println 回显,程序会正常运行,但显然我不想在循环中打印任何内容。

我猜这个问题可能是 Java 编译器试图优化空 while 循环到它停止实际检查“标志”变量的程度的结果?有没有人建议让这项工作正常进行,或者有一些更漂亮的基于并发的方法来让主线程暂停直到 keyPressed 线程执行?

谢谢!

最佳答案

你需要声明标志volatile ,否则编译器可以优化您的代码并跳过标志的读取。

关于java - 当标志在不同线程中更改时循环不会结束,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11595868/

相关文章:

c++ - 将抽象重写方法传递给 Boost::Thread

c++ - 在 win32 中连续串行写入,C++

java - 如何使用velocity模板生成二维码

java 。 JMenu 的 Action 监听器

Java泛型将具体类列表的 future 转换为接口(interface)列表的 future ?

python - 程序/循环仅部分工作

c# - 在嵌套的 while 循环中继续

java - Camera2 - 更改 fragment 时为 "Must be called from main thread of fragment host"

c++ - std::list 线程 push_back、front、pop_front

在 R 中多次运行 'prop.test'