swing界面刷新問題


在Java Swing編程中,往往會遇到需要動態刷新界面的時候,例如動態刷新JLabel的文本,JTextField里的文本等等。但是往往卻沒有達到我們預期的效果,我相信很多朋友都遇到過本文將要說的這個問題。

如下圖的Swing界面中,我們期望在點擊按鈕時,Jlabel和JTextField里的文本能不斷的變化,並實時地顯示出來。

這個例子中,我們期望點擊按鈕后,JLabel和JTextField中每隔一秒鍾刷新一下文本,順序的顯示以下的幾句文本:

復制代碼
Button clicked

Start to change text...

接着顯示數字1到10

action end
復制代碼


很多人都會像下面的代碼這樣實現這個功能:

  1 MainFrame.java
  2 
  3 package com.longyg.test;
  4 
  5 public class MainFrame extends javax.swing.JFrame {
  6 
  7 public MainFrame() {
  8 initComponents();
  9 }
 10 
 11 @SuppressWarnings("unchecked")
 12 // <editor-fold defaultstate="collapsed" desc="Generated Code"> 
 13 private void initComponents() {
 14 
 15 jLabel = new javax.swing.JLabel();
 16 labelText = new javax.swing.JLabel();
 17 jTextField = new javax.swing.JLabel();
 18 fieldText = new javax.swing.JTextField();
 19 button = new javax.swing.JButton();
 20 
 21 setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
 22 
 23 jLabel.setText("JLabel:");
 24 
 25 labelText.setBorder(javax.swing.BorderFactory.createEtchedBorder());
 26 
 27 jTextField.setText("JTextField: ");
 28 
 29 button.setText("click");
 30 button.addActionListener(new java.awt.event.ActionListener() {
 31 public void actionPerformed(java.awt.event.ActionEvent evt) {
 32 buttonActionPerformed(evt);
 33 }
 34 });
 35 
 36 javax.swing.GroupLayout layout = new javax.swing.GroupLayout(getContentPane());
 37 getContentPane().setLayout(layout);
 38 layout.setHorizontalGroup(
 39 layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
 40 .addGroup(layout.createSequentialGroup()
 41 .addGap(10, 10, 10)
 42 .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.TRAILING)
 43 .addComponent(button)
 44 .addGroup(layout.createSequentialGroup()
 45 .addComponent(jLabel)
 46 .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.UNRELATED)
 47 .addComponent(labelText, javax.swing.GroupLayout.PREFERRED_SIZE, 127, javax.swing.GroupLayout.PREFERRED_SIZE))
 48 .addGroup(layout.createSequentialGroup()
 49 .addComponent(jTextField)
 50 .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED)
 51 .addComponent(fieldText, javax.swing.GroupLayout.PREFERRED_SIZE, 127, javax.swing.GroupLayout.PREFERRED_SIZE)))
 52 .addContainerGap(17, Short.MAX_VALUE))
 53 );
 54 layout.setVerticalGroup(
 55 layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
 56 .addGroup(layout.createSequentialGroup()
 57 .addGap(20, 20, 20)
 58 .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
 59 .addComponent(jLabel)
 60 .addComponent(labelText, javax.swing.GroupLayout.PREFERRED_SIZE, 26, javax.swing.GroupLayout.PREFERRED_SIZE))
 61 .addGap(18, 18, 18)
 62 .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
 63 .addComponent(jTextField)
 64 .addComponent(fieldText, javax.swing.GroupLayout.PREFERRED_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.PREFERRED_SIZE))
 65 .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.UNRELATED)
 66 .addComponent(button)
 67 .addContainerGap(javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE))
 68 );
 69 
 70 pack();
 71 }// </editor-fold>
 72 
 73 private void buttonActionPerformed(java.awt.event.ActionEvent evt) { 
 74 changeText("Button clicked");
 75 try {
 76 Thread.sleep(1000);
 77 } catch (InterruptedException ex) {
 78 ex.printStackTrace();
 79 }
 80 changeText("Start to change text...");
 81 try {
 82 Thread.sleep(1000);
 83 } catch (InterruptedException ex) {
 84 ex.printStackTrace();
 85 }
 86 for (int i = 0; i < 10; i++) {
 87 changeText((i+1)+"");
 88 try {
 89 Thread.sleep(1000);
 90 } catch (InterruptedException ex) {
 91 ex.printStackTrace();
 92 }
 93 }
 94 changeText("action end");
 95 }
 96 
 97 private void changeText(String text) {
 98 labelText.setText(text);
 99 fieldText.setText(text);
100 }
101 
102 /**
103 * @param args the command line arguments
104 */
105 public static void main(String args[]) {
106 java.awt.EventQueue.invokeLater(new Runnable() {
107 
108 public void run() {
109 new MainFrame().setVisible(true);
110 }
111 });
112 }
113 // Variables declaration - do not modify 
114 private javax.swing.JButton button;
115 private javax.swing.JTextField fieldText;
116 private javax.swing.JLabel jLabel;
117 private javax.swing.JLabel jTextField;
118 private javax.swing.JLabel labelText;
119 // End of variables declaration 
120 }

 

 

可以看到,在buttonActionPerformed方法中,我們多次調用了setText來期望改變JLabel和JTextField中的文本。當我們運行這段代碼,你會很遺憾的發現,點擊click后,JLabel和JTextField中並沒有如我們所期望的不斷的更新並顯示不同的文本。而是點擊按鈕后,界面仿佛被卡住一樣,等過了一段時間后,顯示出最后一句文本“action end”。

為什么會發生這樣奇怪的現象呢?

Java Swing中,界面刷新是線程同步的,也就是說同一時間,只有一個線程能執行刷新界面的代碼。如果要多次不斷地刷新界面,必須在多線程中調用刷新的方法。

本例中,在buttonActionPerformed方法中多次調用了setText方法來試圖刷新JLabel和JTextField的文本。buttonActionPerformed方法運行在主線程中,所以每次調用setText都是運行在主線程中,而且是順序的執行的。在前面幾次調用setText后,線程並沒有退出,所以界面刷新線程不能獲得執行刷新的機會。而當最后一次setText后,線程退出,界面才能執行刷新。所以我們只能看到最后一次setText的值。

因此,要解決這個問題,我們必須把buttonActionPerformed方法中的代碼段放到一個單獨的線程中執行。這樣它就不會使線程阻塞,當每次setText后,界面刷新線程也能得到執行的機會,從而刷新界面。

下面是修改后的代碼,只有buttonActionPerformed方法的代碼被修改,其他部分的代碼與上面的完全一致。

復制代碼
private void buttonActionPerformed(java.awt.event.ActionEvent evt) {                                       
        new Thread(new Runnable() {
            @Override
            public void run() {
                changeText("Button clicked");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException ex) {
                    ex.printStackTrace();
                }
                changeText("Start to change text...");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException ex) {
                    ex.printStackTrace();
                }
                for (int i = 0; i < 10; i++) {
                    changeText((i+1)+"");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException ex) {
                        ex.printStackTrace();
                    }
                }
                changeText("action end");
            }
        }).start();
    }
復制代碼

我們可以看到,新的buttonActionPerformed方法中,僅僅是把整個代碼段放在了一個線程中,並啟動了線程。

我們在每次setText后,都睡眠了1秒鍾,是為了看到界面真的實時的變化了,如果不睡眠,界面刷新會一閃而過,不利於觀察。

再次運行代碼,會發現,終於得到了我們期望的效果:JLabel和JTextField中的文本動態的變化了!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM