Android開(kāi)發(fā)中關(guān)于獲取當(dāng)前Activity的一些思考
在Android開(kāi)發(fā)過(guò)程中,我們有時(shí)候需要獲取當(dāng)前的Activity實(shí)例,比如彈出Dialog操作,必須要用到這個(gè)。關(guān)于如何實(shí)現(xiàn)由很多種思路,這其中有的簡(jiǎn)單,有的復(fù)雜,這里簡(jiǎn)單總結(jié)一下個(gè)人的一些經(jīng)驗(yàn)吧。
反射
反射是我們經(jīng)常會(huì)想到的方法,思路大概為
- 獲取ActivityThread中所有的ActivityRecord
- 從ActivityRecord中獲取狀態(tài)不是pause的Activity并返回
一個(gè)使用反射來(lái)實(shí)現(xiàn)的代碼大致如下
public static Activity getActivity() {
Class activityThreadClass = null;
try {
activityThreadClass = Class.forName("android.app.ActivityThread");
Object activityThread = activityThreadClass.getMethod("currentActivityThread").invoke(null);
Field activitiesField = activityThreadClass.getDeclaredField("mActivities");
activitiesField.setAccessible(true);
Map activities = (Map) activitiesField.get(activityThread);
for (Object activityRecord : activities.values()) {
Class activityRecordClass = activityRecord.getClass();
Field pausedField = activityRecordClass.getDeclaredField("paused");
pausedField.setAccessible(true);
if (!pausedField.getBoolean(activityRecord)) {
Field activityField = activityRecordClass.getDeclaredField("activity");
activityField.setAccessible(true);
Activity activity = (Activity) activityField.get(activityRecord);
return activity;
}
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
return null;
}
然而這種方法并不是很推薦,主要是有以下的不足:
- 反射通常會(huì)比較慢
- 不穩(wěn)定性,這個(gè)才是不推薦的原因,Android框架代碼存在修改的可能性,誰(shuí)要無(wú)法100%保證mActivities,paused固定不變。所以可靠性不是完全可靠。
Activity基類
既然反射不是很可靠,那么有一種比較可靠的方式,就是使用Activity基類。
在Activity的onResume方法中,將當(dāng)前的Activity實(shí)例保存到一個(gè)變量中。
public class BaseActivity extends Activity{
@Override
protected void onResume() {
super.onResume();
MyActivityManager.getInstance().setCurrentActivity(this);
}
}
然而,這一種方法也不僅完美,因?yàn)檫@種方法是基于約定的,所以必須每個(gè)Activity都繼承BaseActivity,如果一旦出現(xiàn)沒(méi)有繼承BaseActivity的就可能有問(wèn)題。
回調(diào)方法
介紹了上面兩種不是盡善盡美的方法,這里實(shí)際上還是有一種更便捷的方法,那就是通過(guò)Framework提供的回調(diào)來(lái)實(shí)現(xiàn)。
Android自 API 14開(kāi)始引入了一個(gè)方法,即Application的registerActivityLifecycleCallbacks方法,用來(lái)監(jiān)聽(tīng)所有Activity的生命周期回調(diào),比如onActivityCreated,onActivityResumed等。
So,一個(gè)簡(jiǎn)單的實(shí)現(xiàn)如下
public class MyApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {
@Override
public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
}
@Override
public void onActivityStarted(Activity activity) {
}
@Override
public void onActivityResumed(Activity activity) {
MyActivityManager.getInstance().setCurrentActivity(activity);
}
@Override
public void onActivityPaused(Activity activity) {
}
@Override
public void onActivityStopped(Activity activity) {
}
@Override
public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
}
@Override
public void onActivityDestroyed(Activity activity) {
}
});
}
}
然而,金無(wú)足赤人無(wú)完人,這種方法唯一的遺憾就是只支持API 14即其以上。不過(guò)還在現(xiàn)在大多數(shù)設(shè)備都滿足了這個(gè)要求。
為什么是弱引用
可能有人會(huì)帶著疑問(wèn)看到這里,MyActivityManager是個(gè)什么鬼,好,我們現(xiàn)在看一下這個(gè)類的實(shí)現(xiàn)
public class MyActivityManager {
private static MyActivityManager sInstance = new MyActivityManager();
private WeakReference<Activity> sCurrentActivityWeakRef;
private MyActivityManager() {
}
public static MyActivityManager getInstance() {
return sInstance;
}
public Activity getCurrentActivity() {
Activity currentActivity = null;
if (sCurrentActivityWeakRef != null) {
currentActivity = sCurrentActivityWeakRef.get();
}
return currentActivity;
}
public void setCurrentActivity(Activity activity) {
sCurrentActivityWeakRef = new WeakReference<Activity>(activity);
}
}
這個(gè)類,實(shí)現(xiàn)了當(dāng)前Activity的設(shè)置和獲取。
那么為什么要使用弱引用持有Activity實(shí)例呢?
其實(shí)最主要的目的就是避免內(nèi)存泄露,因?yàn)槭褂媚J(rèn)的強(qiáng)引用會(huì)導(dǎo)致Activity實(shí)例無(wú)法釋放,導(dǎo)致內(nèi)存泄露的出現(xiàn)。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家學(xué)習(xí)Android軟件編程有所幫助。
- Android:如何編寫“萬(wàn)能”的Activity
- Android中fragment與activity之間的交互(兩種實(shí)現(xiàn)方式)
- Android activity堆棧及管理實(shí)例詳解
- Android實(shí)現(xiàn)Activity水平和垂直滾動(dòng)條的方法
- Android開(kāi)發(fā)中Activity創(chuàng)建跳轉(zhuǎn)及傳值的方法
- Android實(shí)現(xiàn)在子線程中更新Activity中UI的方法
- 詳解Android開(kāi)發(fā)中Activity的四種launchMode
- Android判斷Activity是否在最上層的方法
- Android編程中activity啟動(dòng)時(shí)出現(xiàn)白屏、黑屏問(wèn)題的解決方法
- Android:“萬(wàn)能”Activity重構(gòu)篇
相關(guān)文章
Android學(xué)習(xí)之Broadcast的簡(jiǎn)單使用
這篇文章主要為大家詳細(xì)介紹了Android學(xué)習(xí)之Broadcast的簡(jiǎn)單使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08
Android 防止多次重復(fù)點(diǎn)擊的三種方法的示例
本篇文章主要介紹了Android 防止多次重復(fù)點(diǎn)擊的三種方法的示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
Android實(shí)現(xiàn)將已發(fā)送的短信寫入短信數(shù)據(jù)庫(kù)的方法
這篇文章主要介紹了Android實(shí)現(xiàn)將已發(fā)送的短信寫入短信數(shù)據(jù)庫(kù)的方法,是Android手機(jī)開(kāi)發(fā)常見(jiàn)的技巧,需要的朋友可以參考下2014-09-09
詳解Android 多級(jí)聯(lián)動(dòng)控件實(shí)現(xiàn)思路討論
這篇文章主要介紹了詳解Android 多級(jí)聯(lián)動(dòng)控件實(shí)現(xiàn)思路討論,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
Android編程實(shí)現(xiàn)實(shí)時(shí)監(jiān)聽(tīng)EditText文本輸入的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)實(shí)時(shí)監(jiān)聽(tīng)EditText文本輸入的方法,結(jié)合實(shí)例形式分析了EditText控件及事件響應(yīng)相關(guān)操作技巧,需要的朋友可以參考下2017-06-06
Android studio gradle環(huán)境變量配置教程
這篇文章主要為大家詳細(xì)介紹了Android studio gradle環(huán)境變量配置教程,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05
Android開(kāi)發(fā) -- UI界面之threme和style
做Java的人一般都做過(guò)CSS,我們都知道它也有一個(gè)樣式,Android中的樣式也可以進(jìn)行類比。2016-06-06

