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

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

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

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

Android

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

Android Studio

Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

Q&A

1回答

3627閲覧

Androidでのボタン処理

退会済みユーザー

退会済みユーザー

総合スコア0

Java

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

Android

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

Android Studio

Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

1グッド

1クリップ

投稿2016/01/31 14:46

###前提・実現したいこと
Androidスマートフォンで、STARTボタンをタップすると2秒後に画面上に加速度センサーの値を表示し、STOPボタンを押すと表示を止める、ということをしようとしています。

ちなみに表示中の値はCSVで出力する予定です。

###発生している問題・エラーメッセージ
STARTボタンを押すと、ちゃんと2秒たってから表示が始まるのですが、
STOPを押しても一瞬止まった後すぐ勝手に表示が再開されてしまいます。

おそらく0にしたbutton_flagが、また勝手に1になってしまっているのだと思いますが、
どうすれば『STARTを押した2秒後に1、STOPを押したら0、そしてそのまま』
というようにできますか?

###ソースコード

JAVA

1 protected void onCreate(Bundle savedInstanceState) { 2 super.onCreate(savedInstanceState); 3 setContentView(R.layout.activity_main); 4 manager = (SensorManager)this.getSystemService(SENSOR_SERVICE); 5 accSensor = this.manager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER); 6 Button start_btn = (Button) findViewById(R.id.button); 7 Button stop_btn = (Button) findViewById(R.id.button2); 8 9 start_btn.setOnClickListener(new View.OnClickListener() { 10 @Override 11 public void onClick(View v) { 12 mHandler.postDelayed(new Runnable() { 13 @Override 14 public void run() { 15 button_flag = 1; 16 mHandler.postDelayed(this, 2000); 17 } 18 },2000); // 2秒後に処理 19 } 20 }); 21 22 stop_btn.setOnClickListener(new View.OnClickListener() { 23 @Override 24 public void onClick(View v) { 25 button_flag = 0; 26 } 27 }); 28 29 mX = (TextView)this.findViewById(R.id.textView); 30 mY = (TextView)this.findViewById(R.id.textView4); 31 mZ = (TextView)this.findViewById(R.id.textView6); 32 33 }

###その他
一部ですが、よろしくお願いします。

yashiromisa👍を押しています

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

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

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

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

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

guest

回答1

0

まず、なぜ勝手に戻るかというと、

  1. STARTでHandlerに渡されるRunnableは、button_flagを1にしたあと、自身を再び2秒後に実行するようHandlerに入れる。
  2. STOPを押したところで前回のRunnableが入れられてから2秒後には再びRunnableが実行され、button_flagを1にして自身をHandlerに入れる。以下同じ動作を続ける。

という動きになっているからです。これを解消するためには、Runnableを変数に保持しておいて、止めるときにHandlerのremoveCallbacks(Runnable)にそれを渡す必要があるのではないでしょうか。

まどろっこしいのでコード全部書き直しました。

java

1protected void onCreate(Bundle savedInstanceState) { 2 super.onCreate(savedInstanceState); 3 setContentView(R.layout.activity_main); 4 manager = (SensorManager)this.getSystemService(SENSOR_SERVICE); 5 accSensor = this.manager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER); 6 Button start_btn = (Button) findViewById(R.id.button); 7 Button stop_btn = (Button) findViewById(R.id.button2); 8 9 final Runnable r = new Runnable(){ 10 @Override 11 public void run() { 12 button_flag = 1; 13 mHandler.postDelayed(this, 2000); 14 } 15 }; 16 17 start_btn.setOnClickListener(new View.OnClickListener() { 18 @Override 19 public void onClick(View v) { 20 mHandler.postDelayed(r ,2000); // 2秒後に処理 21 } 22 }); 23 24 stop_btn.setOnClickListener(new View.OnClickListener() { 25 @Override 26 public void onClick(View v) { 27 //Handlerにセットしている上のRunnableを除去 28 mHandler.removeCallbacks(r); 29 button_flag = 0; 30 } 31 }); 32 33 mX = (TextView)this.findViewById(R.id.textView); 34 mY = (TextView)this.findViewById(R.id.textView4); 35 mZ = (TextView)this.findViewById(R.id.textView6); 36 37}

投稿2016/01/31 15:01

編集2016/01/31 16:22
swordone

総合スコア20649

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

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

退会済みユーザー

退会済みユーザー

2016/01/31 15:20 編集

さっそくの回答、ありがとうございます。 仰る通りに編集しましたところ、後半の『mHandler.removeCallback(r);』の(r)のところにエラーが出ました。『Cannot resolve symbol r』だそうです。 どうしてでしょうか。 教えてください。 追記:変更した部分です。 start_btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { //mHandler.postDelayed(new Runnable() { Runnable r = new Runnable(){ @Override public void run() { button_flag = 1; mHandler.postDelayed(this, 2000); } }; // 2秒後に処理 } }); stop_btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mHandler.removeCallbacks(r); button_flag = 0; } });
swordone

2016/01/31 15:22

失礼しました。 Runnable r の宣言にfinalをつけるか、 メソッドの外に出してメンバ変数にしてください。
退会済みユーザー

退会済みユーザー

2016/01/31 15:30

ありがとうございます。 その部分の記述も教えていただけますか?
swordone

2016/01/31 15:33

finalをつけるパターンで回答を編集しています。
退会済みユーザー

退会済みユーザー

2016/01/31 15:37

大変失礼いたしました。ありがとうございます。 しかしまだエラーが消えないのです。どうしてでしょうか。
swordone

2016/01/31 15:42

では、メンバ変数にする方ではどうでしょうか?
退会済みユーザー

退会済みユーザー

2016/01/31 15:45

メンバ変数もあまりよくわかっていないのですが、 protected void onCreate(Bundle savedInstanceState) { //省略 Runnable r = new Runnable; start_btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mHandler.postDelayed(new Runnable() { @Override public void run() { button_flag = 1; mHandler.postDelayed(this, 2000); } }; } }); で合っていますか? 何度もすみませんが、お願いします。
swordone

2016/01/31 16:23 編集

よく見たらコメントのコードがおかしいですね… 質問で掲載している最初のコードから以下の修正をしてください。 1. start_btn.setOnClickListenerの直前にfinal Runnable r = ~を入れる 2. start_btn.setOnClickListenerの中、onClickでmHandler.postDelayedに渡しているnew Runnable(){~}をrに変える↓(追記:ここ間違えてました) public void onClick(View v) { mHandler.postDelayed(r, 2000); } 3. stop_btn.setOnClickListenerの中、onClickの中にmHandler.removeCallbacks(r);を追加
退会済みユーザー

退会済みユーザー

2016/01/31 16:13

@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); manager = (SensorManager)this.getSystemService(SENSOR_SERVICE); accSensor = this.manager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER); Button start_btn = (Button) findViewById(R.id.button); Button stop_btn = (Button) findViewById(R.id.button2); final Runnable r = new Runnable(); start_btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mHandler.postDelayed(new Runnable() { @Override public void run() { button_flag = 1; mHandler.postDelayed(r, 2000); } }; } }); stop_btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mHandler.removeCallbacks(r); button_flag = 0; } }); mX = (TextView)this.findViewById(R.id.textView); mY = (TextView)this.findViewById(R.id.textView4); mZ = (TextView)this.findViewById(R.id.textView6); } こういうことですか? 2番がよくわかりませんでした。 現在『final Runnable r = new Runnable();』の行の右側と 『mHandler.postDelayed(r, 2000);』の2行下の『};』にエラーが出ています。 前者は『Runnable is abstract;cannot be instantiated』 後者は『")" expected』です。
swordone

2016/01/31 16:19

なぜ私が書いたとおりに書いてくれないのでしょうか? 前者は私が回答で書いたとおりにfinal Runnable r = ~ を宣言すればいいだけです。あなたの元のコードでpostDelayedに渡している匿名クラスのインスタンスをそのまま変数化しているだけです。 後者は、もともと匿名クラスで宣言していたインスタンスを変数化したので、その変数を渡しているだけです。書き方もコメントそのままです。と言うか置き換えている場所が違います。↓をそのまま書いてください。 start_btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mHandler.postDelayed(r ,2000); // 2秒後に処理 } });
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

会員登録して回答してみよう

アカウントをお持ちの方は

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

ただいまの回答率
85.50%

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

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

質問する

関連した質問