我想了解一些线程一次调用swing方法的一些信息。
编辑: 我使用Java 7。
我看到了以下主题:
Thread Safety of JTextArea.append
我开发了一个小型秋千应用程序。
这是我的主类,它是一个线程安全类。 我调用方法SwingUtilities.invokeLater使其成为线程安全类。
MainClass:
package swingex;
import javax.swing.SwingUtilities;
public final class MainClass extends Thread {
public static void main(String[] _args) {
SwingUtilities.invokeLater(new MainClass());
}
@Override
public void run() {
new MainWindow();
}
}
这是一个继承自JFrame的类。 我在内容窗格中放了一个文本区域和一个按钮。
主窗口:
package swingex;
import java.awt.Dimension;
import javax.swing.BoxLayout;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
public final class MainWindow extends JFrame {
private JTextArea textArea = new JTextArea(32,32);
//AddText inherits from Thread class
private AddText thread = new AddText(textArea);
public MainWindow() {
JPanel panel_ = new JPanel();
panel_.setLayout(new BoxLayout(panel_, BoxLayout.PAGE_AXIS));
JScrollPane scr_ = new JScrollPane(textArea);
scr_.setPreferredSize(new Dimension(128, 128));
panel_.add(scr_);
//The button is used for adding rows in the text area
JButton button_ = new JButton("Add rows");
//Adding the event
button_.addActionListener(new AddTextEvent(this));
panel_.add(button_);
setContentPane(panel_);
setDefaultCloseOperation(EXIT_ON_CLOSE);
pack();
setVisible(true);
}
//Called by actionPerformed method
public void setText() {
if (thread.isAlive()) {
//prevent from setting the text area by multi threading
return;
}
//For avoiding issues, the text area is affected by less than two threads.
thread = new AddText(textArea);
thread.start();
}
}
单击按钮使线程休眠5秒钟,然后线程将200行添加到文本区域。 AddText:
package swingex;
import java.awt.Rectangle;
import javax.swing.JTextArea;
import javax.swing.text.BadLocationException;
public final class AddText extends Thread {
private JTextArea textArea;
public AddText(JTextArea _textArea) {
textArea = _textArea;
}
@Override
public void run() {
try {
Thread.sleep(5000);
} catch (InterruptedException _0) {
_0.printStackTrace();
}
//Only one thread can access the code
for (int i = 0; i < 200; i++) {
textArea.append("Text"+i+"\n");
}
int endPosition_ = textArea.getDocument().getLength();
Rectangle bottom_;
try {
bottom_ = textArea.modelToView(endPosition_);
textArea.scrollRectToVisible(bottom_);
} catch (BadLocationException _0) {
_0.printStackTrace();
}
}
}
该类实现ActionListener,用于单击按钮。
AddTextEvent:
package swingex;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
public final class AddTextEvent implements ActionListener {
private MainWindow window;
public AddTextEvent(MainWindow _window) {
window = _window;
}
@Override
public void actionPerformed(ActionEvent _e) {
window.setText();
}
}
提前谢谢。
编辑2:我的“新线程”类如下: (其他类别保持不变。)
package swingex;
import java.awt.Rectangle;
import javax.swing.JTextArea;
import javax.swing.SwingWorker;
import javax.swing.text.BadLocationException;
/**Now the class AddText inherits from SwingWorker*/
public final class AddText extends SwingWorker<Void, Void> {
private JTextArea textArea;
public AddText(JTextArea _textArea) {
textArea = _textArea;
}
/**The equivalent in the class Thread*/
public void start() {
execute();
}
@Override
public Void doInBackground() {
try {
Thread.sleep(5000);
} catch (InterruptedException _0) {
_0.printStackTrace();
}
for (int i = 0; i < 200; i++) {
textArea.append("Text"+i+"\n");
}
int endPosition_ = textArea.getDocument().getLength();
Rectangle bottom_;
try {
bottom_ = textArea.modelToView(endPosition_);
textArea.scrollRectToVisible(bottom_);
} catch (BadLocationException _0) {
_0.printStackTrace();
}
return null;
}
/**The equivalent in the class Thread*/
public boolean isAlive() {
return !isDone() || !isCancelled();
}
}
编辑3:它是一个继承自“swing”组件的自定义类:
我的自定义方法是否“线程安全”?
MyTextArea:
package swingex;
import javax.swing.JTextArea;
public class MyTextArea extends JTextArea {
private String aField = "";
public String getaField() {
return aField;
}
public void setaField(String _aField) {
aField = _aField;
}
}
现在,我使用SwingWorker的“doInBackground”方法中的SwingUtilities方法“invokeLater”
答案 0 :(得分:1)
我认为你应该在Swing的EDT
中采取行动。因此,应该使用invokeLater
调用睡眠后的所有内容。
这不是并发访问的问题。我们不应该在其他线程中修改UI。
你能做的是:
Runnable
中并使用invokeLater
进行调用。然后在EDT保证正确的订单。 对于长时间的操作,请查看swingworker
。
为什么会出现问题? 至少有2个主题: