gpt4 book ai didi

java - 将 Swing 与工作线程以外的线程连接

转载 作者:行者123 更新时间:2023-12-01 23:39:01 25 4
gpt4 key购买 nike

我为一个项目构建了一个简单的客户端和服务器,使用非常基本的文本界面进行测试和开发(下面的代码演示了用法)。

客户端库有一个acceptMessage()方法和一个getMessage()方法,它们基本上分别将消息推送到阻塞队列或从阻塞队列中拉出消息以分别进行输入和输出(在客户端中,这是通过put()和take()实现的)调用)。一个线程在 System.in 上阻塞,并通过acceptMessage 将读取的行发送到客户端,而其他线程则在客户端 getMessage() 方法上阻塞,并在消息到达时将消息回显到 System.out。一切都非常基本,但效果还不错。

现在我已经让我的客户端库可以工作了,我正在尝试弄清楚如何将它集成到使用 Swing GUI 的应用程序中。到目前为止,我除了在 Netbeans 的界面构建工具中构建一个带有文本输入框和标签的简单表单外,还没有做更多的事情。这个想法是让文本输入框代替从 system.in 读取内容,并让标签显示原本写入 system.out 的内容。到那时,我将在 Swing 中复制我的简单测试应用程序。

据我了解,与 Swing GUI 直接交互的所有内容都必须在 Swing 线程中运行,但客户端被构建为作为其自己的线程运行。我不认为从GUI发送消息到acceptMessage()会非常困难(我认为它涉及设置一个ActionPerformed方法,该方法将读取输入框的内容并在客户端上调用acceptMessage(),尽管我'我仍在试图弄清楚),但我不知道如何获取回复。我知道由于线程安全问题,我无法在 GUI 线程中调用客户端功能,而且客户端库的编写方式并不了解其消费类的任何信息。客户端实例只需传递到消费类中,然后使用acceptMessage() 和getMessage() 来发送和接收消息。它不(也不应该)关心消费类实际上是什么。

这种架构是否可以轻松地将客户端集成到 GUI 中?如果是这样,处理客户端输入的正确方法是什么? (就像我说的,我认为一旦我弄清楚了 Swing 的这一方面,向客户端的输出就不会特别困难)。

我使用 Swing 的主要动机是 a) 它似乎是记录最好的 Java GUI 库,b) Netbeans 已经有了使用 Swing 的工具,c) 我有一个严格的截止日期,但没有时间切换 GUI 库并从头开始(这是一个大学项目)。如果我有更多时间,我可能会研究其他库,但我想它们也都有自己的怪癖。

import java.io.*;
import java.util.logging.*;

public class TestApp implements Runnable {

private Client <String> client = null;
private UpstreamChannel upstream = null;
private DownstreamChannel downstream = null;
private Thread upstreamThread = null;
private Thread downstreamThread = null;
private boolean ending = false;

private class UpstreamChannel implements Runnable {

private TestApp outer = null;

@Override
public void run () {

Thread.currentThread ().setName ("TestApp.UpstreamChannel");

try (BufferedReader inReader = new BufferedReader (new InputStreamReader (System.in))) {
while (!this.outer.ending) {
this.outer.client.acceptMessage (inReader.readLine ());
}
} catch (IOException ex) {
Logger.getLogger (this.getClass ().getName ()).log (Level.SEVERE, ex.getMessage (), ex);
} finally {
this.outer.ending = true;
this.outer.downstreamThread.interrupt ();
Thread.currentThread ().interrupt ();
return;
}
}

public UpstreamChannel (TestApp app) {
this.outer = app;
}
}

private class DownstreamChannel implements Runnable {

private TestApp outer = null;

@Override
public void run () {

Thread.currentThread ().setName ("TestApp.DownstreamChannel");

try {
while (!this.outer.ending) {
System.out.println (this.outer.client.getMessage ());
}
} catch (InterruptedException ex) {
Logger.getLogger (this.getClass ().getName ()).log (Level.INFO, ex.getMessage (), ex);
} finally {
this.outer.ending = true;
this.outer.upstreamThread.interrupt ();
Thread.currentThread ().interrupt ();
return;
}
}

public DownstreamChannel (TestApp app) {
this.outer = app;
}
}

@Override
public void run () {
if ((null == this.upstreamThread)
&& (null == this.downstreamThread)) {
this.upstreamThread = new Thread (this.upstream);
this.downstreamThread = new Thread (this.downstream);
this.upstreamThread.start ();
this.downstreamThread.start ();

try {
this.upstreamThread.join ();
this.downstreamThread.join ();
} catch (InterruptedException ex) {
Logger.getLogger (this.getClass ().getName ()).log (Level.INFO, ex.getMessage (), ex);
} finally {
this.upstreamThread.interrupt ();
this.downstreamThread.interrupt ();
System.out.println ("Sayonara");
}
}
}

public TestApp (Client <String> client) {
this.upstream = new UpstreamChannel (this);
this.downstream = new DownstreamChannel (this);
this.client = client;

Logger.getLogger (this.getClass ().getName ()).log (Level.INFO, "Class instantiated");
}
}

启动客户端应用程序的代码如下:

    public static void main (String[] args) throws UnknownHostException, IOException, InterruptedException {
Client <String> clientInstance = new Client ("localhost", 8113);
TestApp app = new TestApp (clientInstance);
Thread clientThread = new Thread (clientInstance);
Thread appThread = new Thread (app);

clientThread.start ();
appThread.start ();

clientThread.join ();
appThread.interrupt ();

System.exit (0);
}
}

编辑:我认为轮询 getMessage (因此会阻塞,直到消息到达)并使用publish() 使其可用的工作线程将是一个解决方案,但我认为存在消息“掉在地板上”的风险“如果两条消息连续快速到达。

    SwingWorker reader                      = new SwingWorker () {
@Override
protected Object doInBackground () throws Exception {
while (!this.isCancelled ()) {
publish (clientInstance.getMessage ());
}
return null;
}
};

最佳答案

所以你的基本问题是如何获得

while (!this.outer.ending) {
System.out.println (this.outer.client.getMessage ());
}

在标签上设置文本而不是打印到标准输出。你是对的,因为这是一个单独的线程,所以你不能直接在 GUI 小部件上调用方法。但是您可以要求 GUI 线程分派(dispatch)一些将在 GUI 线程中运行的代码,

while (!this.outer.ending) {
final String message = this.outer.client.getMessage ();

java.awt.EventQueue.invokeLater(new Runnable() {

public void run() {
theLabel.setText(message);

}
});
}

另一种方法可能就像在 GUI 调用中使用事件处理程序一样简单

  this.outer.client.acceptMessage (textBox.getText());

但是,如果acceptMessage也是一个阻塞调用,那么您还需要一个单独的线程,并让GUI将消息传递给该线程。

关于java - 将 Swing 与工作线程以外的线程连接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18243419/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com