日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android如何获得当前应用显示的Activity

發布時間:2023/12/15 Android 24 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android如何获得当前应用显示的Activity 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

關于獲取當前Activity的一些思考

FEB 21ST, 2016

在Android開發過程中,我們有時候需要獲取當前的Activity實例,比如彈出Dialog操作,必須要用到這個。關于如何實現由很多種思路,這其中有的簡單,有的復雜,這里簡單總結一下個人的一些經驗吧。

反射

反射是我們經常會想到的方法,思路大概為

1 獲取ActivityThread中所有的ActivityRecord?
2 從ActivityRecord中獲取狀態不是pause的Activity并返回

一個使用反射來實現的代碼大致如下

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 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; }

然而這種方法并不是很推薦,主要是有以下的不足:

  • 反射通常會比較慢
  • 不穩定性,這個才是不推薦的原因,Android框架代碼存在修改的可能性,誰要無法100%保證mActivities,paused固定不變。所以可靠性不是完全可靠。

Activity基類

既然反射不是很可靠,那么有一種比較可靠的方式,就是使用Activity基類。

在Activity的onResume方法中,將當前的Activity實例保存到一個變量中。

1 2 3 4 5 6 7 8 public class BaseActivity extends Activity{ @Override protected void onResume() { super.onResume(); MyActivityManager.getInstance().setCurrentActivity(this); } }

然而,這一種方法也不僅完美,因為這種方法是基于約定的,所以必須每個Activity都繼承BaseActivity,如果一旦出現沒有繼承BaseActivity的就可能有問題。

回調方法

介紹了上面兩種不是盡善盡美的方法,這里實際上還是有一種更便捷的方法,那就是通過Framework提供的回調來實現。

Android自 API 14開始引入了一個方法,即Application的registerActivityLifecycleCallbacks方法,用來監聽所有Activity的生命周期回調,比如onActivityCreated,onActivityResumed等。

So,一個簡單的實現如下

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 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即其以上。不過還在現在大多數設備都滿足了這個要求。

為什么是弱引用

可能有人會帶著疑問看到這里,MyActivityManager是個什么鬼,好,我們現在看一下這個類的實現

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 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); } }

這個類,實現了當前Activity的設置和獲取。

那么為什么要使用弱引用持有Activity實例呢?

其實最主要的目的就是避免內存泄露,因為使用默認的強引用會導致Activity實例無法釋放,導致內存泄露的出現。詳細了解弱引用,請參考本文譯文:理解Java中的弱引用



轉載自:http://droidyue.com/blog/2016/02/21/thinking-of-getting-the-current-activity-in-android/

原文有demo可以下載。

總結

以上是生活随笔為你收集整理的Android如何获得当前应用显示的Activity的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。