Android Handler介绍
生活随笔
收集整理的這篇文章主要介紹了
Android Handler介绍
小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
本文轉(zhuǎn)自:http://www.cnblogs.com/keyindex/articles/1822463.html
前言
學(xué)習(xí)android一段時(shí)間了,為了進(jìn)一步了解android的應(yīng)用是如何設(shè)計(jì)開發(fā)的,決定詳細(xì)研究幾個(gè)開源的android應(yīng)用。從一些開源應(yīng)用中吸收點(diǎn)東西,一邊進(jìn)行量的積累,一邊探索android的學(xué)習(xí)研究方向。這里我首先選擇了jwood的Standup Timer 項(xiàng)目。本文將把研究的內(nèi)容筆記整理,建立一個(gè)索引列表。關(guān)鍵詞
Android.os.Handler涉及較多的知識(shí)點(diǎn),我把一些關(guān)鍵詞列舉在下面,將主要介紹Handler:- android.os.Handler 、 android.os.Handler.Callback
- Looper、
- Threadle、Runnable
- Message、Message queue
android.os.Handler
Handler在android里負(fù)責(zé)發(fā)送和處理消息。它的主要用途有: 1)按計(jì)劃發(fā)送消息或執(zhí)行某個(gè)Runnanble(使用POST方法); 2)從其他線程中發(fā)送來的消息放入消息隊(duì)列中,避免線程沖突(常見于更新UI線程) 默認(rèn)情況下,Handler接受的是當(dāng)前線程下的消息循環(huán)實(shí)例(使用Handler(Looper looper)、Handler(Looper looper, Handler.Callback callback)可以指定線程),同時(shí)一個(gè)消息隊(duì)列可以被當(dāng)前線程中的多個(gè)對(duì)象進(jìn)行分發(fā)、處理(在UI線程中,系統(tǒng)已經(jīng)有一個(gè)Activity來處理了,你可以再起若干個(gè)Handler來處理)。在實(shí)例化Handler的時(shí)候,Looper可以是任意線程的,只要有Handler的指針,任何線程也都可以sendMessage。Handler對(duì)于Message的處理不是并發(fā)的。一個(gè)Looper 只有處理完一條Message才會(huì)讀取下一條,所以消息的處理是阻塞形式的(handleMessage()方法里不應(yīng)該有耗時(shí)操作,可以將耗時(shí)操作放在其他線程執(zhí)行,操作完后發(fā)送Message(通過sendMessges方法),然后由handleMessage()更新UI)。倒計(jì)時(shí)程序
利用Timer 編寫一個(gè)倒計(jì)時(shí)程序,程序使用Timer和TimerTask來完成倒計(jì)時(shí),同時(shí)使用sendMessages方法發(fā)送消息,然后在HanleMessage里更新UI。 Activity布局:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="fill_parent"android:layout_height="fill_parent"> <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_gravity="center"android:id="@+id/txt"/> <Buttonandroid:id="@+id/btnStartTime"android:text="開始計(jì)時(shí)"android:layout_width="80dip"android:layout_height="wrap_content" ></Button><Buttonandroid:id="@+id/btnStopTime"android:text="停止計(jì)時(shí)"android:layout_width="80dip"android:layout_height="wrap_content"/><SeekBar android:id="@+id/SeekBar01" android:layout_width="match_parent" android:layout_height="wrap_content"></SeekBar> </LinearLayout>
?
這里使用TextView 來顯示倒計(jì)時(shí)的時(shí)間變化,兩個(gè)按鈕用于控制時(shí)間的開始和停止。SeekBar主要是用于查看線程是否被阻塞(阻塞時(shí)無法拖動(dòng))。
@Overridepublicvoid onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);txt = (TextView) findViewById(R.id.txt);btnStart = (Button) findViewById(R.id.btnStartTime);btnStop = (Button) findViewById(R.id.btnStopTime);Log.d("ThreadId", "onCread:"+ String.valueOf(Thread.currentThread().getId()));myHandler =new Handler(this);btnStart.setOnClickListener(this);btnStop.setOnClickListener(this);}
?
在onCreate方法中初始化元素個(gè)元素,myHandler = new Handler(this); 調(diào)用的是? Handler(Handler.Callback callback)構(gòu)造函數(shù),在回調(diào)方法callback中對(duì)發(fā)送來的消息進(jìn)行處理(這樣我們就不必使用內(nèi)部類的寫法來 重寫HandleMessage()方法了),因此Activity必須實(shí)現(xiàn)android.os.Handler.Callback 接口。我們還在將onCreate 方法的ThreadId 記錄在了Log中用以和消息發(fā)送、處理時(shí)所作的線程進(jìn)行比較。發(fā)送消息@Overridepublicvoid onClick(View v) {switch (v.getId()) {case R.id.btnStartTime:startTimer();break;case R.id.btnStopTime:timer.cancel();break;}}privatesynchronizedvoid startTimer() {timer =new Timer();// TimerTask updateTimerValuesTask = new TimerTask() {// @Override// public void run() {// updateTimerValues();// }//// };//自定義的CallBack模式。Task繼承自TimerTaskTask updateTimerValuesTask =new Task(this);timer.schedule(updateTimerValuesTask, 1000, 1000);}//執(zhí)行耗時(shí)的倒計(jì)時(shí)任務(wù)。 privatevoid updateTimerValues() {total--;Log.d("ThreadId", "send:"+ String.valueOf(Thread.currentThread().getId()));Message msg=new Message();Bundle date =new Bundle();// 存放數(shù)據(jù)date.putInt("time", total);msg.setData(date);msg.what=0;myHandler.sendMessage(msg);//另一種寫法 // Message msg=myHandler.obtainMessage(); // Bundle date = new Bundle();// 存放數(shù)據(jù) // date.putInt("time", total); // msg.setData(date); // msg.what=0; // msg.sendToTarget(); }@Overridepublicvoid TaskRun() {updateTimerValues();}
?
?
實(shí)現(xiàn)Button按鈕的事件處理以此進(jìn)入倒計(jì)時(shí)操作。這里使用的Timer 來執(zhí)行定時(shí)操作(其實(shí)我們完全可以另起一個(gè)線程)。Task類繼承了TimerTask類,里面增加了一個(gè)任務(wù)處理接口來實(shí)現(xiàn)回調(diào)模式,應(yīng)此Activity需要實(shí)現(xiàn)該回調(diào)的接口 ITaskCallBack(這樣做是因?yàn)槲冶容^不喜歡內(nèi)部類的編寫方法)。ICallBack接口和Task類publicinterface ITaskCallBack {void TaskRun(); }publicclass Task extends TimerTask {private ITaskCallBack iTask;public Task(ITaskCallBack iTaskCallBack){super();iTask=iTaskCallBack;}publicvoid setCallBack(ITaskCallBack iTaskCallBack){iTask=iTaskCallBack;}@Overridepublicvoid run() {// TODO Auto-generated method stub iTask.TaskRun();}}
?
?
這是Java的回調(diào)函數(shù)的一般寫法。實(shí)現(xiàn)CallBack/*** 實(shí)現(xiàn)消息處理*/@Overridepublicboolean handleMessage(Message msg) {switch(msg.what){case0:Bundle date=msg.getData();txt.setText(String.valueOf(date.getInt("time")));Log.d("ThreadId", "HandlerMessage:"+ String.valueOf(Thread.currentThread().getId()));Log.d("ThreadId", "msgDate:"+ String.valueOf(date.getInt("time")));break;}returnfalse;}
?
?
可以看到 實(shí)現(xiàn)android.os.Handler.Callback 接口,其實(shí)就是對(duì)handleMessage()方法進(jìn)行重寫(和內(nèi)部類的一個(gè)區(qū)別是,內(nèi)部類的返回值是Void)。運(yùn)行結(jié)果
可以看到在onCreate 方法中線程的ID是1(UI線程) 這與 HandlerMessage 進(jìn)行消息處理時(shí)是所作的線程ID是一樣的,而消息發(fā)送的線程ID則為8非UI線程。使用Threadle進(jìn)行實(shí)現(xiàn)
Activity類publicclass ThreadHandlerrActivity extends Activity implements Callback,OnClickListener {private TextView txt;private Button btnStart, btnStop;private Handler myHandler;private TimerThread timerThread;privateint Total=30;/** Called when the activity is first created. */@Overridepublicvoid onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);txt = (TextView) findViewById(R.id.txt);btnStart = (Button) findViewById(R.id.btnStartTime);btnStop = (Button) findViewById(R.id.btnStopTime);Log.d("ThreadId", "onCread:"+ String.valueOf(Thread.currentThread().getId()));myHandler =new Handler(this);btnStart.setOnClickListener(this);btnStop.setOnClickListener(this);}/*** 實(shí)現(xiàn)消息處理*/@Overridepublicboolean handleMessage(Message msg) {switch(msg.what){case0:Bundle date=msg.getData();txt.setText(String.valueOf(date.getInt("time")));Log.d("ThreadId", "HandlerMessage:"+ String.valueOf(Thread.currentThread().getId()));Log.d("ThreadId", "msgDate:"+ String.valueOf(date.getInt("time")));break;}returnfalse;}@Overridepublicvoid onClick(View v) {switch (v.getId()) {case R.id.btnStartTime://自定義的線程timerThread=new TimerThread(myHandler,60);timerThread.start();break;case R.id.btnStopTime:timerThread.stop();//timerThread.destroy(); break;}}}
?
自定義的線程類*** 自定義的線程類,通過傳入的Handler,和Total 定期執(zhí)行耗時(shí)操作* @author linzijun**/ publicclass TimerThread extends Thread {publicint Total=60;public Handler handler;/*** 初始化構(gòu)造函數(shù)* @param mhandler handler 用于發(fā)送消息* @param total 總周期*/public TimerThread(Handler mhandler,int total){super();handler=mhandler;Total=total;}@Overridepublicvoid run() {while(true){Total--;if(Total<0)break;try {Thread.sleep(1000);} catch (InterruptedException e) {// TODO Auto-generated catch block e.printStackTrace();}Message msg=new Message();Bundle date =new Bundle();// 存放數(shù)據(jù)date.putInt("time", Total);msg.setData(date);msg.what=0;Log.d("ThreadId", "Thread:"+ String.valueOf(Thread.currentThread().getId()));handler.sendMessage(msg);}super.run();}}
?
這里繼承了Thread類,也可以直接實(shí)現(xiàn) Runnable接口。關(guān)于POST
Post的各種方法是把一個(gè)Runnable發(fā)送給消息隊(duì)列,它將在到達(dá)時(shí)進(jìn)行處理。POSTpublicclass PostHandler extends Activity implements OnClickListener, Runnable {private TextView txt;private Button btnStart, btnStop;private Handler myHandler;private Timer timer;privateint total =60;@Overrideprotectedvoid onCreate(Bundle savedInstanceState) {// TODO Auto-generated method stub super.onCreate(savedInstanceState);setContentView(R.layout.main);txt = (TextView) findViewById(R.id.txt);btnStart = (Button) findViewById(R.id.btnStartTime);btnStop = (Button) findViewById(R.id.btnStopTime);Log.d("ThreadId", "onCread:"+ String.valueOf(Thread.currentThread().getId()));myHandler =new Handler(){@Overridepublicvoid handleMessage(Message msg) {switch(msg.what){case0:Bundle date=msg.getData();txt.setText(String.valueOf(date.getInt("time")));Log.d("ThreadId", "HandlerMessage:"+ String.valueOf(Thread.currentThread().getId()));Log.d("ThreadId", "msgDate:"+ String.valueOf(date.getInt("time")));break;}}};btnStart.setOnClickListener(this);btnStop.setOnClickListener(this);}@Overridepublicvoid onClick(View v) {switch (v.getId()) {case R.id.btnStartTime://myHandler.post(this);myHandler.postDelayed(this, 1000);break;case R.id.btnStopTime:break;}}@Overridepublicvoid run() {while(true){total--;if(total<0)break;try {Thread.sleep(1000);} catch (InterruptedException e) {// TODO Auto-generated catch block e.printStackTrace();}Message msg=new Message();Bundle date =new Bundle();// 存放數(shù)據(jù)date.putInt("time", total);msg.setData(date);msg.what=0;Log.d("ThreadId", "POST:"+ String.valueOf(Thread.currentThread().getId()));myHandler.sendMessage(msg);Log.d("ThreadId", "Thread:"+ String.valueOf(Thread.currentThread().getId()));}}}
?
使用POST的方式 是將Runnable 一起發(fā)送給處理的線程(這里為UI),如果Runnable的操作比較耗時(shí)的話那線程將進(jìn)入阻塞狀態(tài)。可以看到先運(yùn)行 Runnable的Run方法 然后在進(jìn)入 HandleMessage() 。我還嘗試了另一種寫法,將TimerThreadPOST過去,運(yùn)行結(jié)果是一樣的。POSTpublicclass PostHandler extends Activity implements OnClickListener, Runnable {private TextView txt;private Button btnStart, btnStop;private Handler myHandler;private Timer timer;privateint total =60;@Overrideprotectedvoid onCreate(Bundle savedInstanceState) {// TODO Auto-generated method stub super.onCreate(savedInstanceState);setContentView(R.layout.main);txt = (TextView) findViewById(R.id.txt);btnStart = (Button) findViewById(R.id.btnStartTime);btnStop = (Button) findViewById(R.id.btnStopTime);Log.d("ThreadId", "onCread:"+ String.valueOf(Thread.currentThread().getId()));myHandler =new Handler(){@Overridepublicvoid handleMessage(Message msg) {switch(msg.what){case0:Bundle date=msg.getData();txt.setText(String.valueOf(date.getInt("time")));Log.d("ThreadId", "HandlerMessage:"+ String.valueOf(Thread.currentThread().getId()));Log.d("ThreadId", "msgDate:"+ String.valueOf(date.getInt("time")));break;}}};btnStart.setOnClickListener(this);btnStop.setOnClickListener(this);}@Overridepublicvoid onClick(View v) {switch (v.getId()) {case R.id.btnStartTime://myHandler.post(this);myHandler.postDelayed(this, 1000);break;case R.id.btnStopTime:break;}}@Overridepublicvoid run() {while(true){total--;if(total<0)break;try {Thread.sleep(1000);} catch (InterruptedException e) {// TODO Auto-generated catch block e.printStackTrace();}Message msg=new Message();Bundle date =new Bundle();// 存放數(shù)據(jù)date.putInt("time", total);msg.setData(date);msg.what=0;Log.d("ThreadId", "POST:"+ String.valueOf(Thread.currentThread().getId()));myHandler.sendMessage(msg);Log.d("ThreadId", "Thread:"+ String.valueOf(Thread.currentThread().getId()));}}}
?
?
可以說POST的各種方法主要是用于 “按計(jì)劃發(fā)送消息或執(zhí)行某個(gè)Runnanble(使用POST方法)”。參考文獻(xiàn)
android學(xué)習(xí)筆記之消息機(jī)制,異步和多線程 android handler概念解釋 SDK轉(zhuǎn)載于:https://www.cnblogs.com/tianyaxue/p/3149118.html
總結(jié)
以上是生活随笔為你收集整理的Android Handler介绍的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: paip.odbc DSN的存储与读取
- 下一篇: 为什么必须是final的呢?