成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

誰動了我的Activity?

網絡 通信技術
不知道大家有沒有想過這樣一個問題,日常開發中最常用到的通過 startActivity() 喚起一個新的 Activity,所創建的 Activity 對象到底被誰持有引用了?新啟動的 Activity 對象在其生命周期中理應是一直被持有引用,不然系統 gc 的時候就會被回收掉,那么其中的引用關系是怎樣的呢?

 [[312428]]

前言

不知道大家有沒有想過這樣一個問題,日常開發中最常用到的通過 startActivity() 喚起一個新的 Activity,所創建的 Activity 對象到底被誰持有引用了?新啟動的 Activity 對象在其生命周期中理應是一直被持有引用,不然系統 gc 的時候就會被回收掉,那么其中的引用關系是怎樣的呢?

為了搞清楚整個問題,筆者便開始了翻找源碼之旅(Android Q),首先得弄清楚 Activity 實例是如何被創建的。

Activity 對象的創建

Activity 的啟動是一個跨進程通信的過程,對客戶端而言,Activity 的創建會回調到ActivityThread 中的 handleLaunchActivity() 方法:

  1. @Override 
  2. public Activity handleLaunchActivity(ActivityClientRecord r, 
  3.       PendingTransactionActions pendingActions, Intent customIntent){ 
  4.   ··· 
  5.   final Activity a = performLaunchActivity(r, customIntent); 
  6.   ··· 
  7.   return a; 

接著在 performLaunchActivity() 方法里找到了 Acitivity 實例的創建:

  1. private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) { 
  2.     ··· 
  3.     ContextImpl appContext = createBaseContextForActivity(r); 
  4.     Activity activity = null
  5.     try { 
  6.       // 注解1:通過 ClassLoader 以及目標 Activity 的類名來創建新的 Activity 實例 
  7.         java.lang.ClassLoader cl = appContext.getClassLoader(); 
  8.       activity = mInstrumentation.newActivity( 
  9.              cl, component.getClassName(), r.intent); 
  10.       ··· 
  11.     } ··· 

Activity 相關的創建工作交由給了 Instrumentation 類處理:

  1. public Activity newActivity(ClassLoader cl, String className, 
  2.       Intent intent) 
  3.       throws InstantiationException, IllegalAccessException, 
  4.       ClassNotFoundException { 
  5.   String pkg = intent != null && intent.getComponent() != null 
  6.               ? intent.getComponent().getPackageName() : null
  7.   return getFactory(pkg).instantiateActivity(cl, className, intent); 

最終的創建工作由進一步交由工廠類 AppComponentFactory 實現:

  1. public @NonNull Activity instantiateActivity(@NonNull ClassLoader cl, @NonNull String className, 
  2.       @Nullable Intent intent) 
  3.       throws InstantiationException, IllegalAccessException, ClassNotFoundException { 
  4.   return (Activity) cl.loadClass(className).newInstance(); 

到這里,Activity 對象的創建過程已經很清晰了:通過 ClassLoader 對象以及類名獲取到目標 Activity 的 Class 對象, 再調用 Class 對象的 newInstance() 方法創建了實例。

用圖形關系表示如下:

 

Activity 對象的引用關系

在清楚了 Activity 對象的創建過程后,讓我們回到一開始的 ActivityThread 的performLaunchActivity() 方法中,接著往下看:

  1. private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) { 
  2.   ··· 
  3.    ContextImpl appContext = createBaseContextForActivity(r); 
  4.    Activity activity = null
  5.    ··· 
  6.   try { 
  7.     Application app = r.packageInfo.makeApplication(false, mInstrumentation); 
  8.     ··· 
  9.     if (activity != null) { 
  10.       ··· 
  11.         activity.attach(appContext, this, getInstrumentation(), r.token, 
  12.           r.ident, app, r.intent, r.activityInfo, title, r.parent, 
  13.              r.embeddedID, r.lastNonConfigurationInstances, config, 
  14.           r.referrer, r.voiceInteractor, window, r.configCallback, 
  15.              r.assistToken); 
  16.        ··· 
  17.         // 注解2:ActivityClientRecord 對象持有 Activity 實例的引用 
  18.       r.activity = activity; 
  19.      } 
  20.       r.setState(ON_CREATE); 
  21.  
  22.     // 注解3:將 ActivityClientRecord 對象添加到 mActivities 集合中 
  23.     synchronized (mResourcesManager) { 
  24.        mActivities.put(r.token, r); 
  25.     } 
  26.  
  27.   } ··· 
  28.  
  29.   return activity; 

在這里,我們似乎找到了想要的答案:

新建的 Activity 對象會被傳進來的 ActivityClientRecord 對象所持有,接著該ActivityClientRecord 對象會被添加到一個名為 mActivities 的集合當中所持有。

ActivityClientRecord 是 ActivityThread 的一個靜態內部類,用于記錄 Activity 相關的信息。其對象的創建過程可以在 LaunchActivityItem 類(Api 28 之后)中找到:

frameworks/base/core/java/android/app/servertransaction/LaunchActivityItem.java:

  1. @Override 
  2. public void execute(ClientTransactionHandler client, IBinder token, 
  3.         PendingTransactionActions pendingActions){ 
  4.   Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "activityStart"); 
  5.   ActivityClientRecord r = new ActivityClientRecord(token, mIntent, mIdent, mInfo, 
  6.       mOverrideConfig, mCompatInfo, mReferrer, mVoiceInteractor, mState, mPersistentState, 
  7.        mPendingResults, mPendingNewIntents, mIsForward, 
  8.        mProfilerInfo, client, mAssistToken); 
  9.    client.handleLaunchActivity(r, pendingActions, null /* customIntent */); 
  10.    Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER); 

再來看一下這個 mActivities 集合:

frameworks/base/core/java/android/app/ActivityThread.java:

  1. ··· 
  2. final ArrayMap<IBinder, ActivityClientRecord> mActivities = new ArrayMap<>(); 
  3. ··· 

mActivities 是一個 map 集合,為 ActivityThread 對象的一個成員變量。既然是一個集合,自然也可以在 Activity 銷毀方法回調中找到移除集合內元素的操作:

  1. /** Core implementation of activity destroy call. */ 
  2. ActivityClientRecord performDestroyActivity(IBinder token, boolean finishing, 
  3.      int configChanges, boolean getNonConfigInstance, String reason){ 
  4.   ActivityClientRecord r = mActivities.get(token); 
  5.   ··· 
  6.   synchronized (mResourcesManager) { 
  7.     mActivities.remove(token); 
  8.   } 
  9.   StrictMode.decrementExpectedActivityCount(activityClass); 
  10.   return r; 

圖形關系表示如下:

 

既然 Activity 的對象是間接被 ActivityThread 對象所持有引用,那么該 ActivityThread 對象理應是單例的形式存在,那么該單例 ActivityThread 對象又是如何被創建以及持有的呢?

ActivityThread 對象的創建

一個新的應用進程創建時,會調用 ActivityThread 的靜態主方法 main(),在這里,我們找到了答案:

frameworks/base/core/java/android/app/ActivityThread.java:

  1. ··· 
  2. // 注解 4:靜態的 ActivityThread 成員變量,用于實現單例 
  3. private static volatile ActivityThread sCurrentActivityThread; 
  4. ··· 
  5.  
  6. // 注解 5: ActivityThread 的主方法入口,由 RuntimeInit 調用 
  7. public static void main(String[] args) { 
  8.     ··· 
  9.     Looper.prepareMainLooper(); 
  10.     ··· 
  11.     // 注解 6: 新建一個 ActivityThread 對象 
  12.     ActivityThread thread = new ActivityThread(); 
  13.     thread.attach(false, startSeq); 
  14.     ··· 
  15.     Looper.loop(); 
  16.  
  17.     throw new RuntimeException("Main thread loop unexpectedly exited"); 
  18. ··· 
  19.  
  20. private void attach(boolean system, long startSeq) { 
  21.     // 注解 7: ActivityThread 對象由靜態成員變量所引用 
  22.     sCurrentActivityThread = this; 
  23.     mSystemThread = system; 
  24.     if (!system) { 
  25.         android.ddm.DdmHandleAppName.setAppName("<pre-initialized>"
  26.                                                 UserHandle.myUserId()); 
  27.         RuntimeInit.setApplicationObject(mAppThread.asBinder()); 
  28.         final IActivityManager mgr = ActivityManager.getService(); 
  29.         try { 
  30.             mgr.attachApplication(mAppThread, startSeq); 
  31.         } catch (RemoteException ex) { 
  32.             throw ex.rethrowFromSystemServer(); 
  33.         } 
  34.         ··· 
  35.     } ··· 

由上面的代碼可知,一個新的應用進程創建時,main() 方法里新建一個 ActivityThread 對象賦予給 ActivityThread 類的一個靜態成員變量 sCurrentActivityThread,從而形成一個應用進程對應一個 ActivityThread 對象(單例) 的關系。

 

總結

每一個新啟動的 Activity,其對象實例通過 Class 類的 newInstance 方法創建后,被包裹在一個 ActivityClientRecord 對象中然后添加到進程唯一的 ActivityThread 對象的成員變量 mActivitys 里。換言之,Activity 對象的持有和釋放都是由 ActivityThread 來管理的。

最后,筆者想額外重申兩點:

源碼中,Activity 對象會在多個方法都有傳遞關系,比較復雜,筆者才疏學淺,可能會漏掉一些別的重要的引用關系沒有分析,歡迎大家指正。

上文的 framework 源碼用的是截稿前最新的 Android Q 版本,不同的 Android 系統版本這部分相關的源碼都會有所改動,不能詳細一一對比分析,望大家見諒。

 

責任編輯:武曉燕 來源: 碼個蛋
相關推薦

2012-12-12 09:56:40

EC2AWSAmazon

2016-10-19 11:00:26

2010-08-26 15:34:12

2023-12-13 10:36:38

Long算法代碼

2021-01-08 09:35:41

LinuxHistory命令

2021-04-19 07:35:01

Linuxhistory命令

2021-04-26 10:24:52

Linux 開發操作系統

2015-06-05 15:47:47

2014-06-11 10:06:09

2010-05-20 09:29:14

谷歌微軟云計算

2015-10-09 11:02:02

2011-01-25 09:24:00

2011-12-30 14:35:20

2011-04-14 13:39:15

jar包

2022-07-25 09:40:41

內存00M

2016-05-04 10:14:32

2020-03-24 14:57:05

戴爾

2017-02-14 14:23:52

大數據春晚

2015-04-17 10:30:13

2017-06-03 16:26:05

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久久亚洲一区 | 国产精品美女久久久久久久久久久 | 午夜国产羞羞视频免费网站 | 久久中文字幕电影 | 国产精品欧美一区二区三区不卡 | 在线视频成人 | 国产91精品久久久久久久网曝门 | 精品久久久久一区二区国产 | 亚洲综合一区二区三区 | 在线观看日韩精品视频 | 夜夜精品浪潮av一区二区三区 | 久久精品一区二区三区四区 | 中文字幕 在线观看 | 欧美精品一区在线 | a级大片免费观看 | 先锋av资源在线 | 天天干天天操天天射 | 色就是色欧美 | 依人成人| 国产一区二区三区四区三区四 | 看a网站 | 99久久国产综合精品麻豆 | 欧美精品一区二区三区在线 | 国产伦精品一区二区三区高清 | 欧美一级片中文字幕 | 亚洲三区在线观看 | 久久国产亚洲 | 中文字幕在线免费视频 | 在线91| 古装人性做爰av网站 | 成人精品鲁一区一区二区 | www.激情.com| 久草网视频| 一区二区在线免费播放 | 久久一热| 国产高清一区二区三区 | 亚洲欧美第一视频 | 国产一区精品在线 | 曰批视频在线观看 | gogo肉体亚洲高清在线视 | 天天干天天玩天天操 |