1. ホーム
  2. android

[解決済み] Android "ビュー階層を作成した元のスレッドだけがそのビューに触れることができます。"

2022-01-30 16:28:30

質問

Androidで簡単な音楽プレーヤーを作りました。各曲のビューにはSeekBarが含まれており、以下のように実装されています。

public class Song extends Activity implements OnClickListener,Runnable {
    private SeekBar progress;
    private MediaPlayer mp;

    // ...

    private ServiceConnection onService = new ServiceConnection() {
          public void onServiceConnected(ComponentName className,
            IBinder rawBinder) {
              appService = ((MPService.LocalBinder)rawBinder).getService(); // service that handles the MediaPlayer
              progress.setVisibility(SeekBar.VISIBLE);
              progress.setProgress(0);
              mp = appService.getMP();
              appService.playSong(title);
              progress.setMax(mp.getDuration());
              new Thread(Song.this).start();
          }
          public void onServiceDisconnected(ComponentName classname) {
              appService = null;
          }
    };

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.song);

        // ...

        progress = (SeekBar) findViewById(R.id.progress);

        // ...
    }

    public void run() {
    int pos = 0;
    int total = mp.getDuration();
    while (mp != null && pos<total) {
        try {
            Thread.sleep(1000);
            pos = appService.getSongPosition();
        } catch (InterruptedException e) {
            return;
        } catch (Exception e) {
            return;
        }
        progress.setProgress(pos);
    }
}

これは問題なく動作します。今度は、曲の進行の秒/分をカウントするタイマーが欲しい。そこで TextView をレイアウトに追加し、それを findViewById()onCreate() で、これを run() の後に progress.setProgress(pos) :

String time = String.format("%d:%d",
            TimeUnit.MILLISECONDS.toMinutes(pos),
            TimeUnit.MILLISECONDS.toSeconds(pos),
            TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(
                    pos))
            );
currentTime.setText(time);  // currentTime = (TextView) findViewById(R.id.current_time);

しかし、最後の行で例外が発生します。

android.view.ViewRoot$CalledFromWrongThreadException: ビュー階層を作成した元のスレッドだけが、そのビューに触れることができます。

しかし、私がここでやっていることは、基本的に SeekBar - でビューを作成します。 onCreate でタッチし、それを run() - で、この苦情は出ません。

解決方法は?

バックグラウンドタスクのうち、UIを更新する部分をメインスレッドに移動させる必要があります。そのための簡単なコードがあります。

runOnUiThread(new Runnable() {

    @Override
    public void run() {

        // Stuff that updates the UI

    }
});

のドキュメント Activity.runOnUiThread .

これをバックグラウンドで動作しているメソッド内にネストし、ブロックの途中に更新を実装するコードをコピーペーストするだけです。できるだけ小さなコードだけを含めるようにします。そうしないと、バックグラウンド・スレッドの目的が達成されなくなります。