`

你构建灵敏的界面了么?

阅读更多
今天帮同学调试程序,发现许多同学初写GUI程序共同的毛病,不能够构建反映灵敏的界面,
并由此导致一些问题的出现。或许今天遇到的问题,再加上Java GUI在循环中调用repaint的问题分析能够对初学Java GUI编程同学有所帮助。
我同学写的程序,把有问题那部分抽离出来,表述为下面一段代码:
package edu.jlu.fuliang;

import java.awt.BorderLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;

public class ProblemTest extends JFrame{
   private JTextArea textArea = new JTextArea(10,10);
   private JButton startButton = new JButton("Start");
     
   public ProblemTest(){
	   JPanel panel = new JPanel();
	   panel.setLayout(new BorderLayout());
	   panel.add(new JScrollPane(textArea),BorderLayout.CENTER);
	   panel.add(startButton,BorderLayout.NORTH);
	   startButton.addActionListener(new ActionListener(){
		public void actionPerformed(ActionEvent e) {
                   ProblemTest.this.append();	
		}
	   });
	  add(panel);
	   setSize(300,300);
	   setVisible(true);
   }
  
   public void append() {
		for (int i = 0; i < 20; i++) {
			textArea.append("Line " + i + "\n");
			try {
				Thread.currentThread().sleep(200);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}  
   public static void main(String[] args) {
	   ProblemTest pt = new ProblemTest();
	   pt.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
   }
}

这段代码本来想作的事情是,当按下开始按钮后,在textArea区域中每隔200毫秒追加
一行信息。但结果却是在textArea中一下打印出来了。
我们分析一下这个问题的原因:
当按下开始按钮时,响应该事件,开始执行
public void actionPerformed(ActionEvent e) {
         ProblemTest.this.append();	
}

开始往textarea中每隔200毫秒追加一行信息,但事实上textarea要想反映出这种变化,肯定要repaint()。但是由于按下开始按钮这个事件还没有响应完,其他通知
textarea重绘的事件在消息队列中排在按下开始按钮这个事件之后,这样当按下开始按钮这个事件响应完毕后,才会响应通知textarea重绘的事件,然而这时候所有的信息都已经追加到textArea中的,所以所有的信息一次全显现了,而不是一条一条显示。
通常解决这个问题的方法是,把追加到textarea的操作放在一个线程中单独去做,在响应
按下开始按钮这个事件的代码中把这个线程开启:
package edu.jlu.fuliang;

import java.awt.BorderLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;

public class ProblemTest extends JFrame{
   private JTextArea textArea = new JTextArea(10,10);
   private JButton startButton = new JButton("Start");
   private Thread thread =  null; 
   
   public ProblemTest(){
	   JPanel panel = new JPanel();
	   panel.setLayout(new BorderLayout());
	   panel.add(new JScrollPane(textArea),BorderLayout.CENTER);
	   panel.add(startButton,BorderLayout.NORTH);
	   startButton.addActionListener(new ActionListener(){
		public void actionPerformed(ActionEvent e) {
               // ProblemTest.this.append();	
			 thread.start();
		}
	   });
	   thread = new Thread(){
			  public void run(){
				  append();
			  }
	   };
	   add(panel);
	   setSize(300,300);
	   setVisible(true);
   }
  
   public void append() {
		for (int i = 0; i < 20; i++) {
			textArea.append("Line " + i + "\n");
			try {
				Thread.currentThread().sleep(200);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}  
   public static void main(String[] args) {
	   ProblemTest pt = new ProblemTest();
	   pt.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
   }
}

细心的同学还会发现第一段代码,按钮按下等到所有的信息一次全显现了的时候才抬起,
这往往是界面不灵敏的表现。
5
4
分享到:
评论
1 楼 yujiang 2008-03-17  
一般的Swing控件都不是线程安全的,
应该在Swing线程里面操作你的控件,
也就是一般应该用 SwingUtilities.invokeLater

相关推荐

Global site tag (gtag.js) - Google Analytics