質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Java

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

Q&A

解決済

3回答

2257閲覧

Handler

daichanman123

総合スコア32

Java

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

0グッド

0クリップ

投稿2016/04/04 06:22

コード public class MainActivity extends Activity{ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); LinearLayout layout = new LinearLayout(this); layout.setOrientation(LinearLayout.VERTICAL); setContentView(layout); final TextView tv=new TextView(this); tv.setText("こんにちは"); layout.addView(tv); Button button = new Button(this); layout.addView(button); button.setText("ボタン"); final Handler handler=new Handler();{ new Thread(new Runnable() { @Override public void run() { handler.post(new Runnable() { @Override public void run() { try{ Thread.sleep(5000); } catch(InterruptedException e){ } tv.setText("別スレッドよりGUIにアクセス"); } }); } }).start(); } } }

上のプログラムなんですが、これを実行すると5秒後にボタンと文字「別スレッドよりGUIにアクセス」が表示されました。
私は実行すると
「こんにちは」とボタンが表示されて5秒後に「別スレッドよりGUIにアクセス」が表示されると思っていたのですが、なぜこのような動作になるのでしょうか?
また
「こんにちは」とボタンが表示されて5秒後に「別スレッドよりGUIにアクセス」が表示される
このようにするにはどこを改善すればいいのでしょうか?
よろしくお願いします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答3

0

なぜこの書き方でボタンが押されてから処理が実行されると思われているのかが気になります。

現在のコードではonCreate内でHandler#startを読んでいるので表示後5秒になっています。
ボタンが押されてからの処理はButton#setInClickListenerでリスナーをセットしないとできません。
該当の処理をOnClicklListener#onClickに記述するとボタンが押されてから5秒後に表示されるようになります。

投稿2016/04/04 06:52

yona

総合スコア18155

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

swordone

2016/04/04 07:01

「ボタンが押されてから」とは言ってないようですよ? 「ボタンが表示されてから」と言っています。
daichanman123

2016/04/04 07:04

ボタンを押すとは言ってませんが... 実行して、 「こんにちは」とボタンが表示されて5秒後に「こんにちは」が上書きされて「別スレッドよりGUIにアクセス」が表示 すると思っていたのですが、 何も表示されずに5秒後にボタンと「別スレッドよりGUIにアクセス」が表示 されたので、どうしてこの結果になったのかという質問です。 ボタンを押しませんのでClickListenerは使わないと思います。
yona

2016/04/04 07:17

見間違えました、ごめんなさい。 理由はHandlerの仕様にあります。 Handlerに与えるRunnableはHandlerがインスタンス化されたスレッドで実行されます。 そのため、今回はUIスレッドでインスタンス化されたHandlerなのでRunnableが実行されます。 質問のコードではThread#sleepを使って5秒間眠らせているためUIスレッドも5秒間眠ってしまいます。 Handler#postDelayedを使うと解決できます。
daichanman123

2016/04/05 00:48

こちらこそわかりにくい言い方ですみません。 ありがとうございます! postDelayedを使うとできました!
guest

0

ベストアンサー

ボタンと文字の描画が5秒後になってしまうのは、
描画処理にたどり着く前にThread.sleep(5000)が実行されてしまっているからだと思われます。

処理を順に見ていくと


layout.addView(tv);
layout.addView(button);

でViewを追加していますが、
この段階では、まだ描画処理等は行われておらず、
onCreate()が終わった後のどこかのタイミングでまとめて描画されたと思います。
詳しくは覚えてないので、興味があったら調べてみてください。


final Handler handler=new Handler();

で、Handlerを生成していますが、
Handlerの引数無しコンストラクタを使用して生成した場合、
その処理を呼んだThread(のLooper)に対するHandlerが生成されます。
ActivityのonCreate()は、UIスレッドから呼ばれることになるので、
このHandlerはUIスレッドに対してのHandlerということになります。


new Thread(...).start();

で、新しいThreadを生成した後、即座に実装してあるrun()が実行されます、
つまり

handler.post(...);

が実行されることになるのですが、
このときのhandlerは先ほど生成したUIスレッドに対するHandlerです。

また、post()メソッドを使用して処理を登録しているため、処理遅延無しでの実行となり、
UIスレッドで処理が実行可能になり次第、登録されたran()が実行されます。

最終的に

try{
Thread.sleep(5000);
}catch(InterruptedException e){
}
tv.setText("別スレッドよりGUIにアクセス");

の処理が、スレッドの生成と実行からほとんど遅延無く実行され、
また、実行されるのはもちろんUIスレッドになるので、
UIスレッドでの処理を約5秒間とめた後、文字を更新するということになります。

UIスレッドでは、先ほど追加したViewの描画等の処理が
全てsleep()でとめられてしまっている状態になっており、
sleep()が終わった後、溜まっていた処理が一気に処理され、
同時に処理されているという具合になったのだと思います。


既に回答されている方がいますが、

Thread.sleep(5000);

の処理を消し、
代わりに

handler.post(...);

handler.postDelayed(..., 5000)

に変更することで、想定通りの動作になると思います。


適当に書いたので補足等あればお願いします。

投稿2016/04/04 07:41

編集2016/04/04 07:43
abs123

総合スコア1280

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

daichanman123

2016/04/05 01:10

詳しい解説ありがとうございました! postDelayedを使うとできました!
guest

0

final Handler handler=new Handler();{

セミコロンで区切られてるので、行末の「{」から後ろはコードブロックとして実行されます。
したがって、onCreateメソッド実行中に5秒間処理停止が走って画面描画が完了するのでは?

投稿2016/04/04 07:11

tkturbo

総合スコア5572

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問