前言
最近項(xiàng)目上有這么一個(gè)需求,實(shí)時(shí)監(jiān)控車(chē)輛信息屈呕,要求每隔3秒鐘刷新一次地圖上的車(chē)輛位置信息微宝。我的想法是先定時(shí)從服務(wù)端獲取數(shù)據(jù)存儲(chǔ)到SharedPreferences中酵熙,然后再定時(shí)從SharedPreferences中獲取數(shù)據(jù)顯示到地圖击喂。對(duì)這個(gè)邏輯我不滿意笨枯,但是一時(shí)也找不到別的方法蚯姆,望大神指教悴务。
在使用定時(shí)任務(wù)的時(shí)候嗅骄,最開(kāi)始想到的是Timer艾杏。無(wú)意中看到一種Handler加Runnable方法凉唐,覺(jué)得還是有必要記錄一下碌冶。
Timer方法
Timer一般結(jié)合TimerTask使用湿痢。先看TimerTask,它是一個(gè)抽象類(lèi)扑庞,里面有一個(gè)run()方法譬重。
public abstract class TimerTask implements Runnable {
......
/**
* The action to be performed by this timer task.
*/
public abstract void run();
......
}
查看TimerTask的源碼,可以看到TimerTask其實(shí)就是實(shí)現(xiàn)了Runnable方法嫩挤,也就是說(shuō)害幅,通過(guò)Timer執(zhí)行定時(shí)任務(wù),其實(shí)就是通過(guò)一個(gè)線程做到的岂昭。
再看Timer以现,它其實(shí)就是一個(gè)線程管理器,通過(guò)schedule方法來(lái)開(kāi)啟一個(gè)線程约啊,并實(shí)現(xiàn)任務(wù)調(diào)度邑遏。Timer工作機(jī)制下篇文章再擼,這里簡(jiǎn)單理解并記錄使用方法恰矩。
/**
* A facility for threads to schedule tasks for future execution in a
* background thread. Tasks may be scheduled for one-time execution, or for
* repeated execution at regular intervals.
......
* <p>After the last live reference to a <tt>Timer</tt> object goes away
* <i>and</i> all outstanding tasks have completed execution, the timer's task
* execution thread terminates gracefully (and becomes subject to garbage
* collection). However, this can take arbitrarily long to occur. By
* default, the task execution thread does not run as a <i>daemon thread</i>,
* so it is capable of keeping an application from terminating. If a caller
* wants to terminate a timer's task execution thread rapidly, the caller
* should invoke the timer's <tt>cancel</tt> method.
......
* <p>This class is thread-safe: multiple threads can share a single
* <tt>Timer</tt> object without the need for external synchronization.
*
* <p>This class does <i>not</i> offer real-time guarantees: it schedules
* tasks using the <tt>Object.wait(long)</tt> method.
......
*/
截幾段源碼注釋
- 當(dāng)所有TimerTask任務(wù)完成记盒,并且Timer引用為空的時(shí)候,會(huì)被GC回收外傅。
- 一般程序退出時(shí)纪吮,TimerTask任務(wù)會(huì)跟隨著終止俩檬,主動(dòng)結(jié)束則用Timer的cancel方法。
- 多個(gè)線程可共用一個(gè)Timer碾盟,也就是多個(gè)TimerTask可以共用一個(gè)Timer棚辽。
- Timer不能保證實(shí)時(shí)任務(wù),所有的任務(wù)都得等待調(diào)度冰肴。
說(shuō)人話屈藐,來(lái)個(gè)比喻。Timer是一個(gè)碼頭大哥(簡(jiǎn)稱(chēng)T老大)熙尉,手底下有一幫小弟(Thread联逻,簡(jiǎn)稱(chēng)w)跟一個(gè)管家(schedule,簡(jiǎn)稱(chēng)S管家)检痰。TimerTask是一個(gè)商人(簡(jiǎn)稱(chēng)K老板)包归,手底下有一幫業(yè)務(wù)經(jīng)理(實(shí)例化的timerTask,負(fù)責(zé)某個(gè)具體的任務(wù)攀细,簡(jiǎn)稱(chēng)p經(jīng)理)箫踩。他有一批貨要在碼頭卸載。于是乎谭贪,K老板找到了T老大境钟。
商人:T老大,我有一批貨要在貴碼頭卸載俭识,能不能幫幫忙慨削,價(jià)錢(qián)好商量!
老大:路過(guò)的都是朋友套媚,好說(shuō)缚态!好說(shuō)!
商人:我這批貨很急堤瘤,能不能立馬就下玫芦?(想要實(shí)時(shí))
老大:先來(lái)后到,這是規(guī)矩氨痉桥帆!能不能馬上卸貨,就要看K老板您的運(yùn)氣了慎皱。要是運(yùn)氣好老虫,就能立馬卸貨,要是運(yùn)氣不好茫多,等個(gè)一年半載都有可能啊祈匙。哈哈哈、、夺欲、還望海涵跪帝!(無(wú)法保證實(shí)時(shí))
商人:那就只能等T老大的好消息了。
商人就只能在碼頭排隊(duì)等候了些阅。某天歉甚,輪到K老板卸貨了。
老大:K老板久等了扑眉!接下來(lái)在下靜候吩咐。
商人:豈敢赖钞!只是鄙人的貨有點(diǎn)雜腰素,需要送到不同的地方,還要麻煩T老大啊雪营。
老大:小事一樁弓千!只管說(shuō)給我的S管家聽(tīng),保證K老板滿意献起!(多個(gè)任務(wù)可共用一個(gè)Timer)
管家:K老板只管吩咐洋访,老夫一定保質(zhì)保量完成。
商人:那就多謝S管家谴餐!具體的任務(wù)我讓我的經(jīng)理們給您匯報(bào)姻政。
得到老板的指示,S管家跟p經(jīng)理們開(kāi)始干活了岂嗓。一個(gè)經(jīng)理代表一個(gè)具體的任務(wù)汁展,一個(gè)w代表一個(gè)線程。
schedule的四個(gè)方法
- schedule(task, date)厌殉,指定時(shí)間執(zhí)行一次
經(jīng)理1:S管家食绿,這批布匹要晚上6點(diǎn)往城東的布衣店送去。
管家:w1公罕,你來(lái)器紧,把這個(gè)送到城東的布衣店去。晚上6點(diǎn)就去楼眷。
- schedule(task, delay)铲汪, 從現(xiàn)在起,delay毫秒后摩桶,執(zhí)行一次
經(jīng)理2:S管家桥状,這批水果過(guò)兩個(gè)小時(shí)往城東的水果店送去。
管家:w2硝清,你來(lái)辅斟,兩個(gè)小時(shí)后,開(kāi)始往城東的水果店送芦拿。
- schedule(task, firstTime, period)士飒,firstTime時(shí)刻開(kāi)始查邢,每隔period毫秒執(zhí)行一次
經(jīng)理3:S管家,這批黃金要明天凌晨4點(diǎn)開(kāi)始酵幕,每隔1小時(shí)往城南的金鋪送一次扰藕。
管家:w3,你來(lái)芳撒,明天早起邓深,凌晨4點(diǎn)開(kāi)始干活,沒(méi)喊停笔刹,就一直干芥备。
- schedule(task, delay, period),現(xiàn)在開(kāi)始舌菜,delay毫秒后萌壳,每隔period毫秒執(zhí)行一次
經(jīng)理4:S管家,這批書(shū)要往城北的陳家送去日月,2個(gè)小時(shí)后他們家才有人袱瓮,每隔一個(gè)小時(shí)送一次。
管家:w4爱咬,你來(lái)尺借,吃點(diǎn)東西,2個(gè)小時(shí)后往陳家送書(shū)精拟,一個(gè)小時(shí)送一次褐望,沒(méi)喊停,就一直干串前。
若干天后瘫里,S管家沒(méi)有收到K老板的支付款,便向T老大匯報(bào)荡碾。
管家:老大谨读,K老板不守信用啊,款項(xiàng)沒(méi)有及時(shí)到賬坛吁,如何處理劳殖?
老大:不守規(guī)矩,以后不跟他玩了拨脉,讓他找別人哆姻。(cancel方法,主動(dòng)結(jié)束timerTask)
商人:那我不得重新找碼頭了玫膀?T老大矛缨,通融通融吧。
老大:沒(méi)門(mén)兒!(一旦取消箕昭,要想繼續(xù)只能重新new一個(gè)Timer)
待K老板交足款項(xiàng)后灵妨,只能重新排隊(duì)等待卸貨。某天落竹,海嘯來(lái)了泌霍,碼頭被毀(程序退出),一切都沒(méi)了(timerTask被動(dòng)結(jié)束)述召。
啰嗦了半天朱转,不知道說(shuō)明白沒(méi)有。還是上代碼积暖,直觀顯示肋拔。
......
private Timer timer = new Timer();
......
private void useTimer() {
//多個(gè)任務(wù)可共享一個(gè)timer
timer.schedule(new MarkerTask(), 1000, 2000);
timer.schedule(new MapTask(), 1000, 3000);
}
......
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.stopRefresh://取消定時(shí)任務(wù)
Log.d("MainActivity", "onClick: stopRefresh");
timer.cancel();
break;
case R.id.continueRefresh://繼續(xù)定時(shí)任務(wù)
Log.d("MainActivity", "onClick: continueRefresh");
Timer timer1 = new Timer();
timer1.schedule(new MarkerTask(), 1000, 2000);
timer1.schedule(new MapTask(), 1, 3000);
break;
default:
break;
}
}
......
private class MapTask extends TimerTask {
@Override
public void run() {
refreshMapViewDetail();
}
}
private class MarkerTask extends TimerTask {
@Override
public void run() {
randomMarkerDetail();
}
}
......
Handler加Runnable方法
這個(gè)方法的核心思想就是在Runnable內(nèi)部,將本runnable繼續(xù)插入主線程隊(duì)列中呀酸。理解了Handler的工作機(jī)制,這個(gè)方法就更好理解琼梆,這里直接貼代碼性誉。
......
mapRefreshRun = new Runnable() {
@Override
public void run() {
//更新地圖上的數(shù)據(jù)
refreshMapViewDetail();
//將本runnable繼續(xù)插入主線程中,再次執(zhí)行茎杂。
handler.postDelayed(this, 3000);
}
};
handler.postDelayed(mapRefreshRun, 1);
......
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.stopRefresh:
Log.d("MainActivity", "onClick: stopRefresh");
handler.removeCallbacks(mapRefreshRun);
break;
case R.id.continueRefresh:
Log.d("MainActivity", "onClick: continueRefresh");
handler.postDelayed(mapRefreshRun, 1);
break;
default:
break;
}
}
......
要取消任務(wù)直接用Handler的removeCallbacks方法错览,要繼續(xù)任務(wù),則將任務(wù)繼續(xù)插入主線程中煌往。
總結(jié)
Timer方法是新建子線程倾哺,在子線程中執(zhí)行想要的動(dòng)作,故不可以直接更改UI刽脖;
Handler方法是在主線程中羞海,插入Runnable,可以直接更改UI曲管。對(duì)界面的操作却邓,比如點(diǎn)擊、滑動(dòng)這些都是要在主線程中排隊(duì)進(jìn)行的院水,如果我們這個(gè)定時(shí)任務(wù)的period太短腊徙,比如設(shè)為0,會(huì)否影響用戶(hù)的操作響應(yīng)速度呢檬某?在這個(gè)簡(jiǎn)單的Demo中撬腾,試了下,沒(méi)有太明顯的感覺(jué)恢恼,但這個(gè)問(wèn)題還是要留意民傻。
本Demo的代碼請(qǐng)移步:GitHub