Java 带有SerialEvent的SwingWorker工作者(附于JTextField)
我正试图用java制作一个serialmonitor,我被卡住了。我尝试将文本(串行输入)添加到JTextArea。我尝试使用SwingWorker,我取得了一些成功,但不够好 我有一个事件(SerialEventListener),用于读取输入数据。 在本例中,我试图将传入数据附加到另一个类中声明的JTextArea,而append方法无法工作。我已经读到,它不应该工作,我需要使用SwingWorker,我做到了。问题是SwingWorker的执行过程只有在我在按钮事件中发出执行命令时才会启动。 总之,我希望在从串行端口获取数据时执行swingworker rutine。 示例:serialevent(到达的数据)->appedrutine(swingworker)->finish 这是我的代码: 保存图形组件的类: 中心板Java 带有SerialEvent的SwingWorker工作者(附于JTextField),java,swing,swingworker,Java,Swing,Swingworker,我正试图用java制作一个serialmonitor,我被卡住了。我尝试将文本(串行输入)添加到JTextArea。我尝试使用SwingWorker,我取得了一些成功,但不够好 我有一个事件(SerialEventListener),用于读取输入数据。 在本例中,我试图将传入数据附加到另一个类中声明的JTextArea,而append方法无法工作。我已经读到,它不应该工作,我需要使用SwingWorker,我做到了。问题是SwingWorker的执行过程只有在我在按钮事件中发出执行命令时才会启动
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.Insets;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.BoxLayout;
import javax.swing.JButton;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTabbedPane;
import javax.swing.JTextArea;
import javax.swing.border.TitledBorder;
public class centerPanel implements ActionListener{
private JTabbedPane tabbed_pane = new JTabbedPane();
public JPanel tab_source = new JPanel();
public JPanel tab_graphical = new JPanel();
private updateData updateText_method;
private JTextArea data_content = new JTextArea();
private JButton Freeze, Clear, Reload;
public centerPanel(){
tabbed_pane.setBorder(new TitledBorder("Data visualization"));
Freeze = new JButton("Freeze");
Freeze.addActionListener(this);
Clear = new JButton("Clear");
Clear.addActionListener(this);
Reload = new JButton("Reload");
Reload.addActionListener(this);
BoxLayout box_layout = new BoxLayout(tab_source,BoxLayout.Y_AXIS);
tab_source.setLayout(box_layout);
data_content.setEditable(false);
data_content.setForeground(Color.WHITE);
data_content.setBackground(Color.DARK_GRAY);
tab_source.add(new JScrollPane(data_content));
JPanel temp_panel = new JPanel(new GridBagLayout());
GridBagConstraints c = new GridBagConstraints();
temp_panel.setBorder(new TitledBorder("Data control"));
temp_panel.setPreferredSize(new Dimension(MainFrame.window_width,(int)(MainFrame.window_height*0.07)));
temp_panel.setMaximumSize(new Dimension(MainFrame.window_width, (int)(MainFrame.window_height*0.12)));
c.gridx = 0;
c.insets = new Insets(0,8,8,0);
temp_panel.add(Freeze,c);
c.gridx = 1;
temp_panel.add(Clear,c);
c.gridx = 2;
temp_panel.add(Reload,c);
tab_source.add(temp_panel,BorderLayout.LINE_END);
tabbed_pane.addTab("Text mode",tab_source);
tabbed_pane.addTab("Graphic mode",tab_graphical);
}
public JTabbedPane getTabs(){
return tabbed_pane;
}
public void updateData(){
updateText_method = new updateData(data_content);
updateText_method.execute();
}
@Override
public void actionPerformed(ActionEvent event) {
if(event.getSource() == Clear){
updateData();
}
}
}
这是我的serialMonitor类(从串行端口读取)
这是我的UpdateData类(SwingWorker所在的位置)
import java.util.List;
导入javax.swing.JTextArea;
导入javax.swing.SwingWorker;
公共类updateData扩展SwingWorker{
私有最终JTextArea消息stextarea;
公共更新数据(最终JTextArea消息stextarea){
this.messagesTextArea=messagesTextArea;
}
@凌驾
受保护的整数doInBackground()引发异常{
int i;
对于(i=0;i你可以做很多工作,但我会给你一些基本知识
最终用户在应用程序中执行的所有操作都在一个线程(事件调度线程(EDT))上运行,除非您有意将其放在另一个线程上。因此,如果用户单击某个按钮,而您开始执行某项需要一段时间的操作,UI将冻结,直到您完成为止
解决这个问题的简单方法是开始一个新的线程来完成你的工作
但是…你还必须考虑在EDT上也必须进行任何UI更新(OH,NOS!)。这样就可以在队列中放置一个进程来运行EDT。
任务是使整个过程变得更简单。它提供了一个接口,用于1)在EDT的线程中执行代码,2)在EDT的线程中执行代码(在第一个线程完成后)
现在,你的问题中有一些不清楚的地方:
serialMonitor
甚至不是代码的一部分-我不确定您想用它做什么
- 不清楚您是希望用户操作触发UI更新,还是串行端口上的事件(通信)应触发更新
如果您的SerialMonitor
已经在自己的线程上执行(而不是在EDT上),我认为它必须执行,那么每当您需要更新UI时,您只需像这样包装调用:
SwingUtilities.invokeLater(new Runnable(){
@Override
public void run() {
//Insert your code to update UI here
}
});
当然,您需要授予SerialMonitor
用户界面更新权限。第一个评论家,遵循java命名代码约定谢谢。我刚刚打印了java代码约定(1997年9月12日)。等待第二个评论家。首先感谢您的评论。主要想法是当用户按“打开”时按钮…串行端口被打开,如果数据从串行端口传入…它将开始更新textarea。因此,我试图做的是仅当我从串行端口获得一些信息时才更新textfield。我想通过在serialMonitor中运行线程来更新UI是可以的。哦…EVRIKA。主要问题是我实例化了我的JTextArea超出了我的构造函数,更大的错误是它不是静态的。事实上,使用invokeLater线程将相关操作插入EDT队列在从串行端口获取大量数据时帮助很大。非常感谢。
import java.util.List;
import javax.swing.JTextArea;
import javax.swing.SwingWorker;
public class updateData extends SwingWorker<Integer, String> {
private final JTextArea messagesTextArea;
public updateData(final JTextArea messagesTextArea) {
this.messagesTextArea = messagesTextArea;
}
@Override
protected Integer doInBackground() throws Exception {
int i;
for ( i = 0; i<50; i++)
publish(new String(""+i));
return 1;
}
@Override
protected void process(final List<String> chunks) {
for (final String string : chunks) {
messagesTextArea.append(string);
messagesTextArea.append("\n");
}
}
SwingUtilities.invokeLater(new Runnable(){
@Override
public void run() {
//Insert your code to update UI here
}
});