2

アプレットでゲームを開発しようとしていますが、この問題が発生しています。ゲームが続行される前に、ユーザーにカウントダウンを表示したいと思います。ただし、カウントダウンは表示されず、代わりに GUI がフリーズします。どうすればこれを回避できますか? この問題を示すコードを次に示します。

編集: 以下のコードは「ほぼ」動作し、タイマーは動作しますが、開始ボタンが押されるたびに画面が新しいタイマー値に更新されるだけです。テキストを自動的に更新するにはどうすればよいですか?

public class TestApplet extends JApplet implements ActionListener{


        final JTextField _displayField = new JTextField("Countdown", 6);
        CountDownTimer clock = new CountDownTimer();
        JButton jbtnStart = new JButton("Start");

    public void addComponentToPane(Container pane) {

        JPanel mainPanel = new JPanel();    
        mainPanel.add(jbtnStart);
        mainPanel.add(_displayField);
        pane.add(mainPanel);
        jbtnStart.addActionListener(this);
    }


  public void init() {

        TestApplet testApplet = new TestApplet();
        testApplet.setVisible(true);    
        testApplet.addComponentToPane(this.getContentPane());   
        this.setSize(200, 100);

}

    public void actionPerformed(ActionEvent e) {

      if ( e.getSource() == jbtnStart   ){
              clock.start(_displayField);
          }
   }     
}

// ********************************************************************************
//********************************************************************************
//********************************************************************************

class CountDownTimer  {

    private static final int N = 60;
    private final ClockListener cl = new ClockListener();
    private final Timer t = new Timer(1000, cl);
    static int count =0;

    public int getCount(){
         System.out.println(count);
        return count;
    }
    public void setCount(int n){
        count = n;
    }

    public CountDownTimer() {
        t.setInitialDelay(0);
    }

    public void start(JTextComponent c) {
        t.start();
       Boolean bool  = false;
          while ( bool ==false){     
              c.setText( "Starting new game in... "+ this.getCount() );
              bool = ( this.getCount()<10 );
          }
    }

    private class ClockListener implements ActionListener {

        public void actionPerformed(ActionEvent e) {
            count %= N;
            count++;
           setCount(count);
        }
    }
}
4

3 に答える 3

4

EDT をブロックしている ActionListener に while ループがあります。表示フィールドを更新するコードは、ActionListener に含めないでください。

代わりに、このコードは Timer クラスにある必要があります。次に、タイマーが起動するたびに、単純に 1 を減算して表示フィールドを更新します。カウントがゼロになると、タイマーを停止します。

また、CountDownTimer は JFrame を拡張しないでください。これは単なるクラスであり、フレームとは関係ありません。

編集:

Swing Timer の簡単な使用法を次に示します。

import java.awt.*;
import java.awt.event.*;
import java.util.*;
import javax.swing.*;

public class TimerTime extends JFrame implements ActionListener
{
    JLabel timeLabel;

    public TimerTime()
    {
        timeLabel = new JLabel( new Date().toString() );
        getContentPane().add(timeLabel, BorderLayout.NORTH);
    }

    public void actionPerformed(ActionEvent e)
    {
        timeLabel.setText( new Date().toString() );
    }

    public static void main(String[] args)
    {
        TimerTime frame = new TimerTime();
        frame.setDefaultCloseOperation( EXIT_ON_CLOSE );
        frame.pack();
        frame.setVisible(true);

        int time = 1000;
        javax.swing.Timer timer = new javax.swing.Timer(time, frame);
        timer.setInitialDelay(1);
        timer.start();
    }
}
于 2013-02-12T16:19:14.983 に答える
1

バックグラウンド スレッドを使用したソリューションが可能かどうかについて議論してきたので、セカンダリ スレッドを使用してソリューションをコーディングしたところ、問題なく動作しました。

  • メソッド「getCount」と「setCount」は不要です
  • 「N」定数はカウントダウンの長さを安定させます

    public class TestApplet extends JApplet implement ActionListener{

    JTextField _displayField;
    CountDownTimer clock;
    JButton jbtnStart;
    Thread thread;
    
    public TestApplet(){
        this.jbtnStart = new JButton("Start");
        this._displayField = new JTextField("Countdown", 30);
        this.clock = new CountDownTimer(_displayField);
        this.thread = null;
    }
    
    public void addComponentToPane(Container pane) {
    
    
    
        JPanel mainPanel = new JPanel();    
        mainPanel.add(jbtnStart);
        mainPanel.add(_displayField);
        pane.add(mainPanel);
        jbtnStart.addActionListener(this);
    }
    
    
    public void init() {
    
        TestApplet testApplet = new TestApplet();
        testApplet.setVisible(true);    
        testApplet.addComponentToPane(this.getContentPane());   
        this.setSize(200, 100);
    
    
    }
    
    public void actionPerformed(ActionEvent e) {
    
        if ( e.getSource() == jbtnStart   ){
    
            if(thread != null){
                thread.interrupt();
            }
            thread = new Thread(clock);
            thread.start();
        }
    }     
    

    }

/ * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ***/ /** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * * / / * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** *** * ** * ** * **** /

public class CountDownTimer implements Runnable{

    private static final int N = 60;
    JTextComponent c;
    static int count =0;

    public int getCount(){
         System.out.println(count);
        return count;
    }
    public void setCount(int n){
        count = n;
    }

    public CountDownTimer(JTextComponent c) {
        this.c = c; 
    }

    @Override
    public void run() {
        try {
            for(int i=N; i>0; i--){
                setCount(i);
                c.setText( "Starting new game in... "+ this.getCount() );
                Thread.sleep(1000); 
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally{
            setCount(0);
        }
    }
}
于 2013-02-13T01:29:46.460 に答える
0

カウントダウンに同時実行を使用するのはどうですか?

        public void actionPerformed(ActionEvent e) {

              if ( e.getSource() == jbtnStart   ){
              //You should start your new thread here
              Thread thread = Thread(new Runnable(){
                 public void run(){
                     clock.start();
                     Boolean bool  = false;
                     while ( bool ==false){
    //You can not modify the UI from a secondary thread, so you should call to some 
    //method in the GUI thread...     
                     _displayField.setText( "Starting new game in... "+ clock.getCount() );
                     bool = ( clock.getCount()==5 );  
                 }

               }
               thread.start();

            }


     }
}  

注: コードは心から行われているため、エラーが発生しやすいですが、要点を表しています。

于 2013-02-12T16:17:55.200 に答える