【问题标题】:Why is starting a thread blocked by another thread?为什么启动一个线程被另一个线程阻塞?
【发布时间】:2013-04-21 02:53:16
【问题描述】:

我在编写架构原型时遇到了一个奇怪的问题。

我尝试创建两个独立调度相同命令的线程。第一个线程是使用Scanner,第二个线程是依赖Swing。问题是第一个线程阻止了第二个线程启动。第二个线程仅在扫描仪获得足够的输入后才开始。强制第一个线程休眠直到第二个线程启动也暂时解决了这个问题。

以下示例非常一致地重现了此行为。在通话之间休眠使其更加一致。

import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.Scanner;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.WindowConstants;

public final class Bug {
    public static void main(final String[] arguments) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("The commands are \"wait\" and \"quit\".");
                final Scanner scanner = new Scanner(System.in);
                loop: while (true) {
                    System.out.print("Enter a command: ");
                    final String command = scanner.nextLine();
                    switch (command.toLowerCase()) {
                    case "exit":
                    case "quit":
                        break loop;
                    default:
                        System.out.println("Use \"wait\" or \"quit\" instead of \"" + command + "\".");
                    case "wait":
                    }
                }
                scanner.close();
            }
        }).start();

        try {
            Thread.sleep(1000);//improves consistency
        }
        catch (final InterruptedException exception) {}

        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                final JFrame frame = new JFrame("Commands");
                frame.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
                frame.setResizable(false);
                frame.setLayout(new BorderLayout());
                frame.add(new JButton("Wait"), BorderLayout.LINE_START);
                final JButton button = new JButton("Quit");
                button.addActionListener(new ActionListener() {
                    @Override
                    public void actionPerformed(final ActionEvent event) {
                        frame.dispose();
                    }
                });
                frame.add(button, BorderLayout.LINE_END);
                frame.pack();
                frame.setVisible(true);
            }
        });
    }
}

为什么第二个线程不能正常启动?是我的错吗?

一个similar problem 是在十年前作为一个错误提交的。


运行java -version 导致

java version "1.7.0_21"
Java(TM) SE Runtime Environment (build 1.7.0_21-b11)
Java HotSpot(TM) Client VM (build 23.21-b01, mixed mode, sharing)

cmd -info

Microsoft Windows XP [Version 5.1.2600]
(C) Copyright 1985-2001 Microsoft Corp.

如果这很重要。

【问题讨论】:

  • 我在您上面发布的代码中看不到任何可以解释您所描述行为的内容。这对我来说是个谜。
  • 为我工作:Ubuntu 13.04; openjdk 1.7.0_21
  • 我在 Windows 7,jdk 1.7 上试过这个代码。有用。你的问题是另外一回事。

标签: java multithreading swing events awt


【解决方案1】:

这段代码对我有用,尽管我不得不用一系列 if/else if 替换开关(我使用的是 Java 1.6)

final String command = scanner.nextLine().toLowerCase();
if (command.equals("exit") || command.equals("quit")) {
    break loop;
} else if (!command.equals("wait")) {
    System.out.println("Use \"wait\" or \"quit\" instead of \"" + command + "\".");
}

【讨论】:

    【解决方案2】:

    这可能是因为您将 EventQueue.invokeLater() 用于第二个线程。当线程启动时,第一个线程可能已经在控制台等待用户输入。我可以想象 EventQueue 可能不会立即触发您的第二个线程,因为第一个线程已经在等待输入,从而阻塞了事件泵。

    我建议您使用不同的机制来启动第二个线程。为什么不像你的第一个线程那样开始呢?

    另外,我同意其他 cmets 的观点,即应该重构此代码,以帮助更清楚地了解发生了什么以及为什么。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-04-27
      • 2012-01-27
      相关资源
      最近更新 更多