1. 程式人生 > >(二)Activity啟動模式

(二)Activity啟動模式

啟動模式

一、概念

顧名思義,就是activity的啟動方式方法,共有4種模式standard,singleTop,singleTask,singleInstance。 為什麼要學習啟動模式呢?因為有些時候我們是需要控制activity的例項數量的,比如通話介面,整個手機中應該只有一個通話activity的例項。再比如瀏覽器,我們可以開啟多個頁面,各個頁面間相互獨立。這類的需求我們就需要使用啟動模式來實現,所以理解每一種啟動模式的特點就很必要了。 這裡插一句,配置activity啟動模式有2種方式,在AndroidManifinest.xml中配置和程式碼配置 接下來就以一個demo為例說明下幾種方式的特點

二、demo相關

2.1 xml佈局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context="com.example.hjw.launchmode.MainActivity">

    <Button
        android:id="@+id/bt_singleInstance"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="bt_singleInstance"
        />
    <Button
        android:id="@+id/bt_singleTop"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="bt_singleTop"
        />
    <Button
        android:id="@+id/bt_singleTask"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="bt_singleTask"
        />
    <Button
        android:id="@+id/bt_self"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="bt_self"
        />

</LinearLayout>

2.2 AndroidManifinest.xml

4個Activity,分別配置4種啟動模式
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="com.example.hjw.launchmode">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity"
            >
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/>

                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>
        </activity>
        <activity
            android:name=".TopActivity"
            android:launchMode="singleTop"
            android:taskAffinity="com.example.hjw.launchmode.top"
            >
            <!--android:taskAffinity="com.example.hjw.launchmode.top"-->

        </activity>
        <activity
            android:name=".TaskActivity"
            android:launchMode="singleTask"
            android:taskAffinity="com.example.hjw.launchmode.task"
            >

        </activity>
        <!---->
        <activity
            android:name=".InstanceActivity"
            android:launchMode="singleInstance"
            android:taskAffinity="com.example.hjw.launchmode.task">

        </activity>
    </application>

</manifest>

2.3 Activity,共有4個Activity,做的事情都是一樣的,這裡就只貼出一個,其他的類似

package com.example.hjw.launchmode;

import android.content.Intent;
import android.os.PersistableBundle;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    public static String TAG = "hjw";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.d(TAG,"onCreate: "+getClass().getSimpleName() + " TaskId: " + getTaskId() + " hasCode:" + this.hashCode());
        findViewById(R.id.bt_singleInstance).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this, InstanceActivity.class);
                startActivity(intent);
            }
        });
        findViewById(R.id.bt_singleTop).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this, TopActivity.class);
                startActivity(intent);
            }
        });
        findViewById(R.id.bt_singleTask).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this, TaskActivity.class);
                startActivity(intent);
            }
        });
        findViewById(R.id.bt_self).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this, MainActivity.class);
                startActivity(intent);
            }
        });

    }

    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        Log.d(TAG,"onCreate: "+getClass().getSimpleName() + " TaskId: " + getTaskId() + " hasCode:" + this.hashCode());
    }
}
好了,準備工作做完了,接下來就進入高潮了

三、4種啟動模式

1,standard

如果沒有指定啟動模式,預設就是這種方式,例子中就是點選self按鈕,啟動自己,看下log
	Line 10264: 01-17 19:28:27.290 D/hjw     ( 5419): onCreate: MainActivity TaskId: 426 hasCode:148911758
	Line 10321: 01-17 19:28:32.445 D/hjw     ( 5419): onCreate: MainActivity TaskId: 426 hasCode:91877479
	Line 10344: 01-17 19:28:33.651 D/hjw     ( 5419): onCreate: MainActivity TaskId: 426 hasCode:81065124
	Line 10366: 01-17 19:28:34.566 D/hjw     ( 5419): onCreate: MainActivity TaskId: 426 hasCode:138172765
結論:每次開啟activity都會建立一個新的例項,點選back的時候是一個一個的銷燬例項。

2,singleTop

(1)這次的順序是先啟動MainActivity,然後一直啟動singleTopActivity,看下log
	Line 9896: 01-17 19:32:46.760 D/hjw     ( 5419): onCreate: MainActivity TaskId: 427 hasCode:56325801
	Line 12280: 01-17 19:32:49.569 D/hjw     ( 5419): onCreate: TopActivity TaskId: 427 hasCode:90947310
	Line 12772: 01-17 19:32:51.491 D/hjw     ( 5419): onNewIntent: TopActivity TaskId: 427 hasCode:90947310
	Line 12780: 01-17 19:32:54.239 D/hjw     ( 5419): onNewIntent: TopActivity TaskId: 427 hasCode:90947310
	Line 12784: 01-17 19:32:54.381 D/hjw     ( 5419): onNewIntent: TopActivity TaskId: 427 hasCode:90947310
可以看到,第一次建立了topActivity,後面都是回撥onNewIntent,複用 (2)這次的順序是,從MainActivity,啟動singleTopActivity,再啟動Main,再啟動幾次Top
	Line 9912: 01-17 19:37:32.003 D/hjw     ( 5817): onCreate: MainActivity TaskId: 429 hasCode:148911758
	Line 10043: 01-17 19:38:06.624 D/hjw     ( 5817): onCreate: TopActivity TaskId: 429 hasCode:47509012
	Line 10086: 01-17 19:38:08.290 D/hjw     ( 5817): onCreate: MainActivity TaskId: 429 hasCode:59946807
	Line 10114: 01-17 19:38:11.327 D/hjw     ( 5817): onCreate: TopActivity TaskId: 429 hasCode:209995572
	Line 10128: 01-17 19:38:12.309 D/hjw     ( 5817): onNewIntent: TopActivity TaskId: 429 hasCode:209995572
	Line 10135: 01-17 19:38:13.131 D/hjw     ( 5817): onNewIntent: TopActivity TaskId: 429 hasCode:209995572
	Line 10151: 01-17 19:38:13.863 D/hjw     ( 5817): onNewIntent: TopActivity TaskId: 429 hasCode:209995572
結論:棧頂複用,只有位於棧頂時,才可以複用

3,singleTask

測試順序採用與singleTop同樣的方式 (1)
	Line 10253: 01-17 19:41:25.604 D/hjw     ( 5817): onCreate: MainActivity TaskId: 430 hasCode:202738547
	Line 13098: 01-17 19:41:28.461 D/hjw     ( 5817): onCreate: TaskActivity TaskId: 431 hasCode:19651321
	Line 13132: 01-17 19:41:30.322 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 431 hasCode:19651321
	Line 13140: 01-17 19:41:30.900 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 431 hasCode:19651321
	Line 13148: 01-17 19:41:31.368 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 431 hasCode:19651321
	Line 13156: 01-17 19:41:31.835 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 431 hasCode:19651321
	Line 13164: 01-17 19:41:32.305 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 431 hasCode:19651321
這個方式下,與singleTop表現一致(細心的同學可能已經發行,TaskId這時候發生了變化,而前邊兩種方式,TaskId沒有發生變化,TaskId這個點留到最後一起說) (2)
	Line 10253: 01-17 19:43:59.360 D/hjw     ( 5817): onCreate: MainActivity TaskId: 432 hasCode:57364092
	Line 10286: 01-17 19:44:03.450 D/hjw     ( 5817): onCreate: TaskActivity TaskId: 433 hasCode:131192636
	Line 10333: 01-17 19:44:06.559 D/hjw     ( 5817): onCreate: MainActivity TaskId: 433 hasCode:157377881
	Line 10358: 01-17 19:44:07.885 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 433 hasCode:131192636
	Line 10385: 01-17 19:44:08.989 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 433 hasCode:131192636
	Line 10393: 01-17 19:44:09.759 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 433 hasCode:131192636
	Line 10401: 01-17 19:44:10.296 D/hjw     ( 5817): onNewIntent: TaskActivity TaskId: 433 hasCode:131192636
這裡就體現出區別了,當再次啟動TaskActivity時,只是呼叫了onNewIntent,說明覆用了該activity。 結論:棧內複用,並且會把它之上的activity例項銷燬。從現象看就是back兩次,棧就空了。當然了還有其他證明方式 adb shell dumpsys activity activities

4,singleInstance


(1)
	Line 10590: 01-17 19:49:49.377 D/hjw     ( 5817): onCreate: MainActivity TaskId: 434 hasCode:23450184
	Line 10616: 01-17 19:49:51.074 D/hjw     ( 5817): onCreate: InstanceActivity TaskId: 435 hasCode:177643169
	Line 10633: 01-17 19:49:52.268 D/hjw     ( 5817): onNewIntent: InstanceActivity TaskId: 435 hasCode:177643169
	Line 10641: 01-17 19:49:52.489 D/hjw     ( 5817): onNewIntent: InstanceActivity TaskId: 435 hasCode:177643169
	Line 10649: 01-17 19:49:52.759 D/hjw     ( 5817): onNewIntent: InstanceActivity TaskId: 435 hasCode:177643169
這個表現與singleTask一樣 (2)
	Line 10170: 01-17 19:52:58.799 D/hjw     ( 5817): onCreate: MainActivity TaskId: 436 hasCode:119769668
	Line 10213: 01-17 19:53:01.271 D/hjw     ( 5817): onCreate: InstanceActivity TaskId: 437 hasCode:94725757
	Line 10238: 01-17 19:53:03.052 D/hjw     ( 5817): onCreate: MainActivity TaskId: 436 hasCode:91870454
	Line 10293: 01-17 19:53:05.881 D/hjw     ( 5817): onNewIntent: InstanceActivity TaskId: 437 hasCode:94725757
	Line 10307: 01-17 19:53:08.611 D/hjw     ( 5817): onNewIntent: InstanceActivity TaskId: 437 hasCode:94725757
	Line 10325: 01-17 19:53:09.233 D/hjw     ( 5817): onNewIntent: InstanceActivity TaskId: 437 hasCode:94725757
結論:建立一個新的task,task只放這一個例項,複用
至此,4種啟動模式的特點就介紹完了。 還有一個遺留的問題,TaskId,就是指任務棧的編號可以通過在AndroidManifinest.xml中給Activity設定taskAffinity屬性。 activity在啟動的時候會優先尋找自己指定的TaskId,對於前2種啟動方式,這個屬性並不生效,後兩種,是有效的。至於為啥,沒研究過,請大神指點。另外我自己還有一個問題
	Line 10443: 01-17 19:58:51.761 D/hjw     ( 5817): onCreate: MainActivity TaskId: 438 hasCode:6413022
	Line 10467: 01-17 19:58:54.475 D/hjw     ( 5817): onCreate: TopActivity TaskId: 438 hasCode:6067614
	Line 10495: 01-17 19:58:56.345 D/hjw     ( 5817): onCreate: InstanceActivity TaskId: 439 hasCode:222214387
	Line 10530: 01-17 19:58:57.975 D/hjw     ( 5817): onCreate: TopActivity TaskId: 440 hasCode:180213716
	Line 10548: 01-17 19:58:59.457 D/hjw     ( 5817): onNewIntent: TopActivity TaskId: 440 hasCode:180213716
應該從log中也能看出順序,先Main,再Top,再Instance,再Top,這個時候可以看到,Top被放到440中了,而不是之前的438中。求大神指點