Android之定时器实现的几种方式和removeCallbacks失效问题详解
轉(zhuǎn)載:http://blog.csdn.net/xiaanming/article/details/9011193
實(shí)現(xiàn)定時器有很多種方式,在這里我簡單的介紹幾種方式
實(shí)現(xiàn)定時器有很多種方式,在這里我簡單的介紹幾種方式
(1)使用Handler + Runnable的方式
Handler handler = new Handler(); Runnable runnable = new Runnable() { @Override public void run() { //你要做的事 //...... System.out.println(Thread.currentThread().getName()); handler.postDelayed(runnable, 1000); } };然后調(diào)用handler.post(runnable);就能啟動定時器,這里是每隔1s打印線程名字,從打印中我們可以知道,他并沒有另開線程,而是運(yùn)行在UI線程當(dāng)中,當(dāng)你要取消定時器的時候,只需要調(diào)用handler.removeCallbacks(runnable)就可以了。
上面中有一個問題,有時候你會發(fā)現(xiàn)removeCallbacks有時候會失效,不能從消息隊列中移除,看下面的demo
圖:兩個按鈕,一個將Runnable加到消息隊列中,一個將Runnable從消息隊列中移除。該Runnable每1秒鐘打印一次日志。 import android.app.Activity; import android.os.Bundle; import android.os.Handler; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; public class TimerActivity extends Activity{ Handler handler = new Handler(); Runnable runnable = new Runnable() { @Override public void run() { System.out.println("update..."); handler.postDelayed(runnable, 1000); } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.timer); Button mButtonStart = (Button) findViewById(R.id.button1); Button mButtonStop = (Button) findViewById(R.id.button2); mButtonStart.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { handler.post(runnable); } }); mButtonStop.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { handler.removeCallbacks(runnable); } }); } } 結(jié)果:(1)start –> ?輸出 –> stop–> 停止輸出
(2)start –> 輸出 –> ?Background –> Front –> stop->繼續(xù)輸出
當(dāng)Activity進(jìn)入后臺運(yùn)行后再轉(zhuǎn)入前臺運(yùn)行,removeCallbacks無法將updateThread從message queue中移除。
這是為什么呢?
在Activity由前臺轉(zhuǎn)后臺過程中,線程是一直在運(yùn)行的,但是當(dāng)Activity轉(zhuǎn)入前臺時會重新定義Runnable runnable;也就是說此時從message queue移除的runnable與原先加入message queue中的runnable并非是同一個對象。如果把runnable定義為靜態(tài)的則removeCallbacks不會失效,對于靜態(tài)變量在內(nèi)存中只有一個拷貝(節(jié)省內(nèi)存),JVM只為靜態(tài)分配一次內(nèi)存,在加載類的過程中完成靜態(tài)變量的內(nèi)存分配,我們做如下修改就能解決上面的這個問題
上面的每一秒打印語句,run方法是運(yùn)行在子線程,不能直接在里面更新UI操作,這里需要注意下,取消的話調(diào)用timer.cancel()就能移除任務(wù)了
(3)采用Handle與線程的sleep(long )方法
1.定義一個Handler類,用于處理接受到的Message
Handler handler = new Handler() { public void handleMessage(Message msg) { super.handleMessage(msg); System.out.println("update..."); }2.新建一個實(shí)現(xiàn)Runnable接口的線程類,用一個boolean 來控制線程開始和結(jié)束 ?boolean isLive = true如下: public class MyThread implements Runnable { @Override public void run() { while (isLive) { try { Thread.sleep(1000);// 線程暫停1秒,單位毫秒 Message message = new Message(); message.what = 1; handler.sendMessage(message);// 發(fā)送消息 } catch (InterruptedException e) { e.printStackTrace(); } } } }
3.在需要啟動線程的地方加入下面語句 new Thread(new MyThread()).start();4.取消的話將isLive設(shè)置為false就行了
總結(jié)
以上是生活随笔為你收集整理的Android之定时器实现的几种方式和removeCallbacks失效问题详解的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Android之如何用cmd方法查看lo
- 下一篇: Android之进程与线程的讲解