Android handler 詳解(面試必問)
handler在Android中被稱為“消息處理者”,在多線程中比較常用。
Handler為Android提供了一種異步消息處理機制,當(dāng)向消息隊列中發(fā)送消息 (sendMessage)后就立即返回,而從消息隊列中讀取消息時會阻塞,其中從消息隊列中讀取消息時會執(zhí)行Handler中的public void handleMessage(Message msg) 方法,因此在創(chuàng)建Handler時應(yīng)該使用匿名內(nèi)部類重寫該方法,在該方法中寫上讀取到消息后的操作,使用Handler的 obtainMessage() 來獲得消息對象。
Handler與線程的關(guān)系:
使用Handler的post方法將Runnable對象放到Handler的線程隊列中后,該Runnable的執(zhí)行其實并未單獨開啟線程,而是仍然在當(dāng)前Activity線程中執(zhí)行的,Handler只是調(diào)用了Runnable對象的run方法。
Bundle是什么:
Bundle是一個特殊的map,它是傳遞信息的工具,它的鍵只能是string類型,而且值也只能是常見的基本數(shù)據(jù)類型。
handler內(nèi)部實現(xiàn)原理

handler實現(xiàn)機制:
1,Message對象,表示要傳遞的一個消息,內(nèi)部使用鏈表數(shù)據(jù)結(jié)構(gòu)實現(xiàn)一個消息池,用于重復(fù)利用,
避免大量創(chuàng)建消息對象,造成內(nèi)存浪費
2,MessageQueue對象,存放消息對象的消息隊列,先進(jìn)先出原則
3,Looper對象負(fù)責(zé)管理當(dāng)前線程的消息隊列
4,handler對象負(fù)責(zé)把消息push到消息隊列中,以及接收Looper從消息隊列中取出的消息
handler的內(nèi)存泄露問題(activity已經(jīng)退出,而handler沒有退出,這樣就可能導(dǎo)致內(nèi)存泄露)
1,定義一個內(nèi)部類時,會默認(rèn)擁有外部類對象的引用,所以建議使用內(nèi)部類時,最好定義一個靜態(tài)內(nèi)部類
2,引用的強弱:強引用→軟引用→弱引用
下面用獲取網(wǎng)絡(luò)圖片代碼說明一下handler基本使用
package com.example.uri;
import java.io.IOException;
import java.io.InputStream;
import java.lang.ref.WeakReference;
import java.net.MalformedURLException;
import java.net.URL;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.app.Activity;
import android.content.Context;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.view.Menu;
import android.view.View;
import android.widget.ImageView;
/**
*
* 訪問網(wǎng)絡(luò)的操作,必須放在工作線程中完成
*
*/
public class MainActivity extends Activity {
private static final int LOADSUCCESS=0x1;
private static ImageView iv;
private final myhandler handler=new myhandler(this);
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
iv=(ImageView) findViewById(R.id.imageView1);
}
private static class myhandler extends Handler{
private final WeakReference<MainActivity> weakReference;
public myhandler(MainActivity mainActivity){
weakReference=new WeakReference<MainActivity>(mainActivity);
}
public void handleMessage(Message msg) {
MainActivity mainActivity=weakReference.get();
if(mainActivity!=null){
switch (msg.what) {
case LOADSUCCESS:
MainActivity.iv.setImageBitmap((Bitmap) msg.obj);
break;
}
}
}
}
public void geturl(View v){
/*Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse("http://www.baidu.com"));
startActivity(intent); */
new Thread(new Runnable() {
@Override
public void run() {
try {
URL url=new URL("http://img2.3lian.com/img2007/10/28/123.jpg");
InputStream in=url.openStream();
Bitmap bitmap=BitmapFactory.decodeStream(in);
Message message= handler.obtainMessage(LOADSUCCESS, bitmap);
handler.sendMessage(message);
} catch (MalformedURLException e) {
// TODO 自動生成的 catch 塊
e.printStackTrace();
} catch (IOException e) {
// TODO 自動生成的 catch 塊
e.printStackTrace();
}
}
}).start();
}
}
如何讓Handler執(zhí)行Runnable時打開新的線程:
1、首先生成一個HandlerThread對象,實現(xiàn)了使用Looper來處理消息隊列的功能,這個類由Android應(yīng)用程序框架提供
HandlerThread handlerThread = new HandlerThread("handler_thread");
2、在使用HandlerThread的getLooper()方法之前,必須先調(diào)用該類的start();
handlerThread。start();
3、根據(jù)這個HandlerThread對象得到其中的Looper對象。
4、創(chuàng)建自定義的繼承于Handler類的子類,其中實現(xiàn)一個參數(shù)為Looper對象的構(gòu)造方法,方法內(nèi)容調(diào)用父類的構(gòu)造函數(shù)即可。
5、使用第三步得到的Looper對象創(chuàng)建自定義的Handler子類的對象,再將消息(Message)發(fā)送到該Handler的消息隊列中,Handler復(fù)寫的handleMessage()將會執(zhí)行來處理消息隊列中的消息。
以上給大家詳細(xì)介紹了Android handler 相關(guān)知識,希望對大家有所幫助!
相關(guān)文章
Android實現(xiàn)創(chuàng)意LoadingView動畫效果
這篇文章主要介紹了Android實現(xiàn)創(chuàng)意LoadingView動畫效果的相關(guān)資料,需要的朋友可以參考下2016-02-02
Android使用ViewStub實現(xiàn)布局優(yōu)化方法示例
這篇文章主要為大家介紹了Android使用ViewStub實現(xiàn)布局優(yōu)化方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
Android 使用viewpager實現(xiàn)無限循環(huán)(定時+手動)
這篇文章主要介紹了Android 使用viewpager實現(xiàn)無限循環(huán)(定時+手動)的相關(guān)資料,需要的朋友可以參考下2015-11-11
使用ListView實現(xiàn)網(wǎng)上訂餐首頁
這篇文章主要為大家詳細(xì)介紹了使用ListView實現(xiàn)網(wǎng)上訂餐首頁,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-01-01

