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

