關(guān)于Activity你所需要知道內(nèi)容(一)

根據(jù)下面的目錄來(lái)介紹和理解Activity中的知識(shí)點(diǎn):

1.Activity的生命周期
2.Activity的緩存方法
3.Activity啟動(dòng)模式
4.Activity顯式隱式啟動(dòng)
5.Activity的加載過(guò)程

Activity的生命周期

1.通過(guò)下面的一張圖片來(lái)理解正常情況下的Activity的生命周期

image.png

根據(jù)上述的圖片可以簡(jiǎn)單了解Activity的生命周期。 下面介紹具體各個(gè)方法:
onCreate(): Activity生命周期中的第一個(gè)方法,在這里做一些初始化。
onStart(): 這個(gè)時(shí)候Activity已經(jīng)可以看見(jiàn),剛剛呈現(xiàn)出來(lái)。
onResume(): Activity完全呈現(xiàn)在用戶面前,可以和用戶進(jìn)行交互。
onPause(): Activity還可以看見(jiàn),但是失去了焦點(diǎn),無(wú)法交互。比如一個(gè)透明的Activity在前臺(tái)。
onStop(): Activity不可見(jiàn),可以做一些稍微重量級(jí)的回收工作。
onDestroy(): Activity被銷毀,可以做一些回收工作和最終資源的銷毀。
onRestart(): 回退到生命棧中原來(lái)的Activity就會(huì)走這個(gè)onRestart()。

2.有兩個(gè)Activity分別為A,B,當(dāng)A啟動(dòng)B時(shí)的生命周期如下:

08-17 13:42:04.666 14674-14674/com.example.administrator.androidartdevelop I/ActivityA: onCreate: 
08-17 13:42:04.679 14674-14674/com.example.administrator.androidartdevelop I/ActivityA: onStart: 
08-17 13:42:04.682 14674-14674/com.example.administrator.androidartdevelop I/ActivityA: onResume: 
08-17 13:42:31.200 14674-14674/com.example.administrator.androidartdevelop I/ActivityA: onPause: 
08-17 13:42:31.239 14674-14674/com.example.administrator.androidartdevelop I/ActivityB: onCreate: 
08-17 13:42:31.240 14674-14674/com.example.administrator.androidartdevelop I/ActivityB: onStart: 
08-17 13:42:31.241 14674-14674/com.example.administrator.androidartdevelop I/ActivityB: onResume: 
08-17 13:42:31.744 14674-14674/com.example.administrator.androidartdevelop I/ActivityA: onStop: 

所有也驗(yàn)證了onPause中少做耗時(shí)的回收操作,這樣會(huì)影響下一個(gè)Activity的啟動(dòng)顯示。

Activity的緩存方法

  1. 所謂的的緩存方法其實(shí)就是異常情況下的Activity的生命周期。
image.png

以下兩種情況才會(huì)有走onSaveInstanceState和onRestoreInstanceState的步驟:

  • 資源相關(guān)的系統(tǒng)配置方式改變,導(dǎo)致Activity被銷毀后重新創(chuàng)建。(比如橫豎的切換,當(dāng)然也可以避免重建,下面我會(huì)講)
  • 由于內(nèi)存太低,殺死優(yōu)先級(jí)較低的Activity。

其實(shí)UI控件都會(huì)有緩存方法,比如TextView中的源碼中就有onSaveInstanceState和onRestoreInstanceState的方法,來(lái)存儲(chǔ)TextView的內(nèi)容。一般在Activity中會(huì)如下使用

 protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.i(TAG, "onCreate: ");
        if(savedInstanceState != null){
            name = savedInstanceState.getString(NAME);
            textView.setText(name);
        }
    }

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        outState.putString(NAME,"pg one");
        Log.i(TAG, "onSaveInstanceState: ");
    }

    @Override
    protected void onRestoreInstanceState(Bundle savedInstanceState) {
        super.onRestoreInstanceState(savedInstanceState);
        Log.i(TAG, "onRestoreInstanceState: name = " + savedInstanceState.getString(NAME));
    }
  1. 通過(guò)指定Activity的configChanges屬性可以避免重新創(chuàng)建。其中比較常用的屬性有
    local(設(shè)備的本地位置發(fā)生了改變,一般指切換了系統(tǒng)語(yǔ)言)。
    orientation(屏幕方向發(fā)生改變)。
    keyboardHidden(鍵盤(pán)的可訪問(wèn)性發(fā)生改變,比如用戶調(diào)出了鍵盤(pán))。
    screenSize(當(dāng)屏幕尺寸方式改變,如果minSdkVersion和targetSdkVersion均小于13,那么不會(huì)導(dǎo)致Activity重啟,否則會(huì)重啟)。
    smallestSreenSize(設(shè)備的物理尺寸,比如接外部設(shè)備,如果minSdkVersion和targetSdkVersion均小于13,那么不會(huì)導(dǎo)致Activity重啟,否則會(huì)重啟)
    避免橫豎屏切換導(dǎo)致Activity重置,只需在Activity中添加android:configChanges="orientation|screenSize"即可

Activity的啟動(dòng)模式

  1. Activity的啟動(dòng)模式分別為standard,singleTop,singleTask,singleInstance。

standard啟動(dòng)模式下:每次都新建Activity放入啟動(dòng)它的Activity所在的任務(wù)棧中


image.png

singleTop啟動(dòng)模式下:如果任務(wù)棧頂部有需要啟動(dòng)的Activity,則不再重新創(chuàng)建,走onNewInstance()

image.png

singleTask啟動(dòng)模式下:下面的流程圖基本上已經(jīng)把所有的情況包含進(jìn)來(lái)。

  • 當(dāng)需要啟動(dòng)的Activity的任務(wù)棧不存在的情況下,創(chuàng)建任務(wù)棧,然后創(chuàng)建該Activity,然后入棧
  • 當(dāng)需要啟動(dòng)的Activity的任務(wù)棧存在,該任務(wù)棧中沒(méi)有該Activity,則創(chuàng)建該Activity,然后入棧
  • 當(dāng)需要啟動(dòng)的Activity的任務(wù)棧存在,該任務(wù)棧中有該Activity,則清空該Activity以上的activity來(lái)到棧頂
image.png

singleInstance啟動(dòng)模式下:
這種啟動(dòng)模式比較特殊,因?yàn)樗鼤?huì)啟用一個(gè)新的棧結(jié)構(gòu),將Activity放置于這個(gè)新的棧結(jié)構(gòu)中,并保證不再有其他Activity實(shí)例進(jìn)入。
我們?cè)O(shè)置FirstActivity的launchMode=”standard”,SecondActivity的launchMode=”singleInstance”
我們發(fā)現(xiàn)這兩個(gè)Activity實(shí)例分別被放置在不同的棧結(jié)構(gòu)中,關(guān)于singleInstance的原理圖如下


singleInstance.png

我們看到從MainActivity跳轉(zhuǎn)到SecondActivity時(shí),重新啟用了一個(gè)新的棧結(jié)構(gòu),來(lái)放置SecondActivity實(shí)例,然后按下后退鍵,再次回到原始棧結(jié)構(gòu);圖中下半部分顯示的在SecondActivity中再次跳轉(zhuǎn)到MainActivity,這個(gè)時(shí)候系統(tǒng)會(huì)在原始棧結(jié)構(gòu)中生成一個(gè)MainActivity實(shí)例,然后回退兩次,注意,并沒(méi)有退出,而是回到了SecondActivity,為什么呢?是因?yàn)閺腟econdActivity跳轉(zhuǎn)到MainActivity的時(shí)候,我們的起點(diǎn)變成了SecondActivity實(shí)例所在的棧結(jié)構(gòu),這樣一來(lái),我們需要“回歸”到這個(gè)棧結(jié)構(gòu)。

  1. Activity的Flags有很多,有的標(biāo)記位可以設(shè)定Activity的啟動(dòng)模式,比如FLAG_ACTIVITY_NEW_TASK(singleTask)和FLAG_ACTIVITY_SINGLE_TOP(singleTop)等。

Activity的顯式隱式啟動(dòng)

1.兩種啟動(dòng)方式:
顯式啟動(dòng)

  Intent intent = new Intent(ActivityA.this,ActivityB.class);
  startActivity(intent);

隱式啟動(dòng)

Intent intent = new Intent();
intent.setAction("com.test.action");
intent.addCategory("com.test.category");
intent.setDataAndType(Uri.parse("content://abc"),"image/*");
startActivity(intent);

2.顯式啟動(dòng)很直接明了,就不多做介紹。隱式啟動(dòng)相對(duì)復(fù)雜,主要是action,category,data等匹配。下面是AndroidManifest.xml中的一個(gè)Activity的intent- filter

<activity android:name=".Chapter_01_Activity.ActivityB">
    <intent-filter>
         <action android:name="com.test.action"/>
         <action android:name="com.test.action1"/>
         <category android:name="android.intent.category.DEFAULT"/>
         <category android:name="com.test.category"/>
         <data
            android:scheme="http"
            android:host="www.test.com"
            android:port="80"
            android:path="/a"
            android:mimeType="image/*"/>
         <data android:mimeType="video/*"/>
    </intent-filter>
</activity>

action的匹配方式:intent中必須存在一個(gè)action,并且action要可以匹配到intent-filter中的action。setAction方法會(huì)覆蓋之前的賦值。也就是說(shuō)最后一次setAction的內(nèi)容必須可以匹配intent-filter中的action。
category的匹配方式:intent中可以沒(méi)有category,因?yàn)橄到y(tǒng)會(huì)默認(rèn)添加android.intent.category.DEFAULT,所以intent-filter中category一定要添加。如果intent中有多個(gè)category,那么必須每個(gè)category都必須在intent-filter中的category配對(duì)到。
data的匹配方式:與action類似。
data由 URI和mimeType兩個(gè)部分組成。URI的組成結(jié)構(gòu)如下:[scheme]://[host]:[port]|[path]|[pathPrefix][pathPattern]
上述代碼中的URI組成如下:http://www.test.com:80/a

3.隱式啟動(dòng)往往需要在啟動(dòng)之前可檢驗(yàn)是否有Activity符合過(guò)濾規(guī)則,不檢驗(yàn)容易出現(xiàn)ActivityNotFoundException的錯(cuò)誤。需要在啟動(dòng)前加入以下代碼:

PackageManager packageManager = getPackageManager();
List<ResolveInfo> activities = packageManager.queryIntentActivities(intent, 0);
boolean isIntentSafe = activities.size() > 0;
if (isIntentSafe) {
   startActivity(intent);
}

Activity的加載過(guò)程

整個(gè)啟動(dòng)流程如下所示:

image.png
  • startActivity有很多重載方法,最后都會(huì)調(diào)用Activity.startActivityForResult()
  • mParent表示ActivityGroup,但是在API13以后ActivityGroup已經(jīng)被棄用,由Fragment代替。因此mParent為null。進(jìn)入Instrumentation.execStartActivity方法
  • 在execStartActivity中調(diào)用ActivityManagerNative.getDefault().startActivity方法。實(shí)際是AMS的Binder對(duì)象執(zhí)行startActivity。
  • 然后依次調(diào)用ActivityStackSupervisor中startActivityMayWait方法,startActivityLocked方法startActivityUncheckedLocked方法。
  • 接著依次調(diào)用ActivityStack的resumeTopActivitiesLocked方法,resumeTopActivityInnerLocked方法。
  • 又回到ActivityStackSupervisor中調(diào)用startSpecificActivityLocked方法,realStartActivityLocked方法。
  • 再調(diào)用ApplicationThread.scheduleLaunchActivity()方法,作用是發(fā)送一個(gè)啟動(dòng)Activity的消息有Handler處理
  • ActivityThread.handleLauchActivity()方法接收發(fā)送過(guò)來(lái)的信息,其內(nèi)部的performLauchActivity來(lái)實(shí)現(xiàn)Activity的創(chuàng)建和啟動(dòng)。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容