web-dev-qa-db-ja.com

RunnableおよびpostDelayedを使用したUIの更新がタイマーアプリで機能しない

私はこれを機能させるために見つけることができるすべての議論とスレッドを見てきましたが、そうではありません。テキストビューを更新する単純なタイマーがあります(以下の例ではmTimeTextField)。 mUpdateTimeTask runメソッドは正しく(毎秒)実行されていますが、UI /テキストフィールドは更新されていません。

ここにある情報に基づいたコードがあります:

http://Android-developers.blogspot.com/2007/11/stitch-in-time.htmlhttp://developer.Android.com/resources/articles/timed- ui-updates.html

これがコードです:

_package com.something.handlertest;

import com.something.handlertest.R;

import Android.app.Activity;
import Android.content.Intent;
import Android.os.Bundle;
import Android.os.Handler;
import Android.os.SystemClock;
import Android.view.View;
import Android.widget.Button;
import Android.widget.TextView;

public class Test extends Activity {

    private Handler mHandler = new Handler(); 
    private int labelNo    = 0;
    private long currTime  = 0L;
    private long mStartTime = 0L;
    TextView mTimeTextField;

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        mTimeTextField = (TextView)findViewById(R.id.timeTextFieldl);

        Button startButton = (Button)findViewById(R.id.start_button);
        startButton.setOnClickListener(new View.OnClickListener() {

            public void onClick(View v) {
                   if (mStartTime == 0L) {
                        mStartTime = System.currentTimeMillis();
                        mHandler.removeCallbacks(mUpdateTimeTask);
                        mHandler.postDelayed(mUpdateTimeTask, 100);
                   }
            }
        });
    }

    private Runnable mUpdateTimeTask = new Runnable() {
           public void run() {
               final long start = mStartTime;
               //long millis = SystemClock.uptimeMillis() - start;
               long millis = SystemClock.uptimeMillis();
               int seconds = (int) (millis / 1000);
               int minutes = seconds / 60;
               seconds     = seconds % 60;

               //setContentView(mTimeTextField);  This will blow up if I use it

               if (seconds < 10) {
                   mTimeTextField.setText("" + minutes + ":0" + seconds);
               } else {
                   mTimeTextField.setText("" + minutes + ":" + seconds);            
               }

               //mHandler.postAtTime(this,
                   //    start + (((minutes * 60) + seconds + 1) * 1000));

               mHandler.postAtTime(this, 1000);
           }
        };

}
_

いくつかの提案に従って、私は追加してみました:

_setContentView(mTimeLabel);
_

しかし、これはビューが親を持たないことに不満を言うでしょう。ちなみに、私は持っています:

_setContentView(R.layout.main);
_

onCreate()を呼び出します。

22
bdelliott

交換する

mHandler.postAtTime(this, 1000);

mHandler.postDelayed(this, 1000);
19
Fedor
  1. UIスレッドでUIの更新を行う必要があります。
  2. UIスレッドでタスクを実行するには、UIスレッドでハンドラーを作成する必要があります。

    public class MainActivity extends AppCompatActivity
        private Handler mHandler;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            mHandler = new Handler(); // This to create the Handler in the UI thread
            // ...
        }
    
5
david m lee

これはあなたが探しているものに沿っているかもしれません:

private Runnable mUpdateTimeTask = new Runnable() {
       public void run() {

           final long start = mStartTime;
           long elapseTime = System.currentTimeMillis() - start;
           int seconds = (int) (elapseTime / 1000);
           int minutes = seconds / 60;
           seconds     = seconds % 60;

           if (seconds < 10) {
               mTimeTextField.setText("" + minutes + ":0" + seconds);
           } else {
               mTimeTextField.setText("" + minutes + ":" + seconds);            
           }

           // add a delay to adjust for computation time
           long delay = (1000 - (elapseTime%1000));

           mHandler.postDelayed(this, delay);
       }
    };

これはビデオの時間を増やす良い方法だと思いました

4
Laith Alnagem

ビューはUIスレッドでのみ表示できます。 ruを使用してランナブルを起動する必要があります

1

R.id.timeTextFieldlが正しいIDであることを確認してください。

0
Fedor
if (seconds < 10) {
    mTimeTextField.setText("" + minutes + ":0" + seconds);
} else {
    mTimeTextField.setText("" + minutes + ":" + seconds);            
}

より短いです:

mTimeTextField.setText(minutes + ":" + (seconds < 10 ? "0" : "") + seconds);

しかし、より良い方法はString.format ^を使用することです

mTimeTextField.setText(String.format("%d:%s%d", minutes, (seconds < 10 ? "0" : ""), seconds);
0
bask0xff