【轉】Android 定時器實現的幾種方式和removeCallbacks失效問題詳解--不錯

來源:互聯網
上載者:User

標籤:

原文網址:http://blog.csdn.net/xiaanming/article/details/9011193

實現定時器有很多種方式,在這裡我簡單的介紹幾種方式

(1)使用Handler + Runnable的方式

 

[java] view plaincopy 
  1. Handler handler = new Handler();  
  2. Runnable runnable = new Runnable() {  
  3.       
  4.     @Override  
  5.     public void run() {  
  6.         //你要做的事  
  7.         //......  
  8.         System.out.println(Thread.currentThread().getName());  
  9.         handler.postDelayed(runnable, 1000);  
  10.     }  
  11. };  

 

然後調用handler.post(runnable);就能啟動定時器,這裡是每隔1s列印線程名字,從列印中我們可以知道,他並沒有另開線程,而是運行在UI線程當中,當你要取消定時器的時候,只需要調用handler.removeCallbacks(runnable)就可以了。

上面中有一個問題,有時候你會發現removeCallbacks有時候會失效,不能從訊息佇列中移除,看下面的demo

圖:兩個按鈕,一個將Runnable加到訊息佇列中,一個將Runnable從訊息佇列中移除。該Runnable每1秒鐘列印一次日誌。

 

[java] view plaincopy 
  1. <span style="font-family:Courier New;">package com.example.demoactivity;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.os.Handler;  
  6. import android.view.View;  
  7. import android.view.View.OnClickListener;  
  8. import android.widget.Button;  
  9.   
  10. public class TimerActivity extends Activity{  
  11.     Handler handler = new Handler();  
  12.     Runnable runnable = new Runnable() {  
  13.           
  14.         @Override  
  15.         public void run() {  
  16.             System.out.println("update...");  
  17.             handler.postDelayed(runnable, 1000);  
  18.         }  
  19.     };  
  20.   
  21.     @Override  
  22.     protected void onCreate(Bundle savedInstanceState) {  
  23.         super.onCreate(savedInstanceState);  
  24.         setContentView(R.layout.timer);  
  25.           
  26.         Button mButtonStart = (Button) findViewById(R.id.button1);  
  27.         Button mButtonStop = (Button) findViewById(R.id.button2);  
  28.           
  29.         mButtonStart.setOnClickListener(new OnClickListener() {  
  30.               
  31.             @Override  
  32.             public void onClick(View v) {  
  33.                 handler.post(runnable);  
  34.             }  
  35.         });  
  36.           
  37.         mButtonStop.setOnClickListener(new OnClickListener() {  
  38.               
  39.             @Override  
  40.             public void onClick(View v) {  
  41.                 handler.removeCallbacks(runnable);  
  42.             }  
  43.         });  
  44.     }  
  45.       
  46. }</span><span style="font-family: Georgia, ‘Times new roman‘, Times, san-serif;">  
  47. </span>  

結果:
(1)start –>  輸出 –> stop–> 停止輸出
(2)start –> 輸出 –>  Background –> Front –> stop->繼續輸出

 

當Activity進入後台運行後再轉入前台運行,removeCallbacks無法將updateThread從message queue中移除。
這是為什麼呢?
在Activity由前台轉後台過程中,線程是一直在啟動並執行,但是當Activity轉入前台時會重新定義Runnable runnable;也就是說此時從message queue移除的runnable與原先加入message queue中的runnable並非是同一個對象。如果把runnable定義為靜態則removeCallbacks不會失效,對於靜態變數在記憶體中只有一個拷貝(節省記憶體),JVM只為靜態分配一次記憶體,在載入類的過程中完成靜態變數的記憶體配置,我們做如下修改就能解決上面的這個問題

 “在Activity由前台轉後台過程中,線程是一直在啟動並執行,但是當Activity轉入前台時會重新定義Runnable runnable”應該不會重新定義Runnable runnable。

[java] view plaincopy 
  1. static Handler handler = new Handler();  
  2. static Runnable runnable = new Runnable() {  
  3.       
  4.     @Override  
  5.     public void run() {  
  6.         System.out.println("update...");  
  7.         handler.postDelayed(runnable, 1000);  
  8.     }  
  9. };  

(2)使用Timer的方式

 

 

[java] view plaincopy 
  1. Timer timer = new Timer();  
  2. timer.schedule(new TimerTask() {  
  3.                       
  4.     @Override  
  5.     public void run() {  
  6.         System.out.println("update....");  
  7.     }  
  8. }, 0, 1000);  

上面的每一秒列印語句,run方法是運行在子線程,不能直接在裡面更新UI操作,這裡需要注意下,取消的話調用timer.cancel()就能移除任務了

 

(3)採用Handle與線程的sleep(long )方法

1.定義一個Handler類,用於處理接受到的Message

 

[java] view plaincopy 
  1. Handler handler = new Handler() {  
  2.         public void handleMessage(Message msg) {  
  3.             super.handleMessage(msg);  
  4.             System.out.println("update...");  
  5.         }  
  6.     }  

2.建立一個實現Runnable介面的線程類,用一個boolean 來控制線程開始和結束  boolean isLive = true如下:

 

 

[java] view plaincopy 
  1. public class MyThread implements Runnable {  
  2.         @Override  
  3.         public void run() {  
  4.             while (isLive) {  
  5.                 try {  
  6.                     Thread.sleep(1000);// 線程暫停1秒,單位毫秒  
  7.                     Message message = new Message();  
  8.                     message.what = 1;  
  9.                     handler.sendMessage(message);// 發送訊息  
  10.                 } catch (InterruptedException e) {  
  11.                     e.printStackTrace();  
  12.                 }  
  13.             }  
  14.         }  
  15.     }  

 

3.在需要啟動線程的地方加入下面語句

 

 

[java] view plaincopy 
  1. new Thread(new MyThread()).start();  

4.取消的話將isLive設定為false就行了

 

今天主要介紹這三種方法,寫的不好的地方希望大家指出,謝謝!

【轉】Android 定時器實現的幾種方式和removeCallbacks失效問題詳解--不錯

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.