Activity 启动模式

4 种启动模式

Android 是以 的形式管理 Activity的,当前活动的Activity位于顶。
通过给Activity指定启动模式Activity设定了在中的管理方式。
Activity有 4 种启动模式,通过AndroidManifest.xml<activity>标签的 android:launchMode 属性来设定启动模式

standard

Activity使用了 standard 模式,每启动一个该Activity,系统都会创建一个新的该Activity的实例并放到顶,不论中原先是否存在该Activity的实例。
standardActivity的默认启动模式,在没有指定启动模式的情形下,Activity 自动使用该模式。

singleTask

Activity使用了 singleTask 模式,每次启动该Activity,系统都会检查中原先是否存在该Activity的实例。如果中不存在该Activity的实例,那么就创建一个新的该Activity的实例;如果中存在该Activity的实例,那么就直接使用该实例,同时把中处于该Activity实例之上的其他Activity实例都出栈。

singleTop

singleTop模式与 singleTask 模式有一定区别。
Activity使用了 singleTop 模式,每次启动该Activity,系统都会检查顶是否是该Activity的实例。如果顶不是该Activity的实例,那么就创建一个新的该Activity的实例;如果顶是该Activity的实例,那么就直接使用该实例。

singleInstance

指定为 singleInstance 模式的Activity,会启用一个新(独立的)来管理该Activity
这样做可以方便不同应用进程共享一个Activity实例。

代码演示

让我们编写代码,运行演示,看看这几种启动模式有什么差异吧。

编写测试代码
  1. 在项目 ActivitySample中创建一个类,名为 BasicSingleActivity,继承 AppCompatActivity.

Java 代码:

public class BasicSingleActivity extends AppCompatActivity {
}

Kotlin 代码:

open class BasicSingleActivity : AppCompatActivity(){
}
  1. 依次创建 4 个Activity,分别是 StandardActivitySingleTaskActivitySingleTopActivitySingleInstanceActivity,使它们都继承 BasicSingleActivity.

Java 代码:

public class StandardActivity extends BasicSingleActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }
}

public class SingleTaskActivity extends BasicSingleActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }
}

public class SingleTopActivity extends BasicSingleActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }
}

public class SingleInstanceActivity extends BasicSingleActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }
}

Kotlin 代码:

class StandardActivity : BasicSingleActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }
}

class SingleTaskActivity : BasicSingleActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }
}

class SingleTopActivity : BasicSingleActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }
}

class SingleInstanceActivity : BasicSingleActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }
}
  1. 配置 AndroidManifest.xml 清单文件,注册新创建的 4 个Activity
    分别为它们配置一个相应的启动模式。
        <activity
            android:name=".SingleInstanceActivity"
            android:exported="true"
            android:launchMode="singleInstance" />
        <activity
            android:name=".SingleTopActivity"
            android:exported="true"
            android:launchMode="singleTop" />
        <activity
            android:name=".SingleTaskActivity"
            android:exported="true"
            android:launchMode="singleTask" />
        <activity
            android:name=".StandardActivity"
            android:exported="true"
            android:launchMode="standard" />
  1. 创建布局文件 activity_basic_single.xml
    编辑布局内容,添加了 4 个按钮,用于启动对应的 Activity
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center_horizontal"
    android:orientation="vertical"
    android:padding="30dp">

    <Button
        android:id="@+id/standardBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="standard"
        android:textAllCaps="false" />
    <Button
        android:id="@+id/singleTaskBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="singleTask"
        android:textAllCaps="false" />
    <Button
        android:id="@+id/singleTopBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="singleTop"
        android:textAllCaps="false" />
    <Button
        android:id="@+id/singleInstanceBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="singleInstance"
        android:textAllCaps="false" />
</LinearLayout>
  1. 加载布局文件,初始化设置
    编辑 BasicSingleActivity 代码,加载布局,初始化。

Java 代码:

public class BasicSingleActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_basic_single);
        Log.i("launchMode_Activity", String.format("onCreate() task: %d, instance: %s", getTaskId(), this.toString()));

        Button standardBtn = findViewById(R.id.standardBtn);
        standardBtn.setOnClickListener(view -> {
            Intent intent = new Intent(this, StandardActivity.class);
            startActivity(intent);
        });
        Button singleTaskBtn = findViewById(R.id.singleTaskBtn);
        singleTaskBtn.setOnClickListener(view -> {
            Intent intent = new Intent(this, SingleTaskActivity.class);
            startActivity(intent);
        });
        Button singleTopBtn = findViewById(R.id.singleTopBtn);
        singleTopBtn.setOnClickListener(view -> {
            Intent intent = new Intent(this, SingleTopActivity.class);
            startActivity(intent);
        });
        Button singleInstanceBtn = findViewById(R.id.singleInstanceBtn);
        singleInstanceBtn.setOnClickListener(view -> {
            Intent intent = new Intent(this, SingleInstanceActivity.class);
            startActivity(intent);
        });
    }
}

Kotlin 代码:

open class BasicSingleActivity : AppCompatActivity(){
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_basic_single)
        Log.i("launchMode_Activity","onCreate() task: $taskId, instance: ${this.toString()}")

        val standardBtn: Button = findViewById(R.id.standardBtn)
        standardBtn.setOnClickListener{
            val intent = Intent(this, StandardActivity::class.java)
            startActivity(intent)
        }
        val singleTaskBtn: Button = findViewById(R.id.singleTaskBtn)
        singleTaskBtn.setOnClickListener{
            val intent = Intent(this, SingleTaskActivity::class.java)
            startActivity(intent)
        }
        val singleTopBtn: Button = findViewById(R.id.singleTopBtn)
        singleTopBtn.setOnClickListener{
            val intent = Intent(this, SingleTopActivity::class.java)
            startActivity(intent)
        }
        val singleInstanceBtn: Button = findViewById(R.id.singleInstanceBtn)
        singleInstanceBtn.setOnClickListener{
            val intent = Intent(this, SingleInstanceActivity::class.java)
            startActivity(intent)
        }
    }
}

这里重写了 onCreate()函数,加载布局,日志记录当前 Activity所在 id以及当前的Activity实例;为 4 个按钮设置事件监听,分别启动对应的 Activity

  1. 编辑项目主 Activity的布局文件 activity_my1.xml
    增加一个按钮,用于启动新建的 Activity
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center_horizontal"
    android:orientation="vertical"
    android:padding="30dp">

    ......

    <Button
        android:id="@+id/standardBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="standard"
        android:textAllCaps="false" />
</LinearLayout>
  1. 编辑项目的主 Activity,设置事件监听
    编辑 MyActivity1,为新增的按钮设置点击事件,跳转至 StandardActivity

Java 代码:

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my1);

        ......

        Button standardBtn = findViewById(R.id.standardBtn);
        standardBtn.setOnClickListener(view -> {
            Intent intent = new Intent(this, StandardActivity.class);
            startActivity(intent);
        });
    }

Kotlin 代码:

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_my1)

        ......

        val standardBtn: Button = findViewById(R.id.standardBtn)
        standardBtn.setOnClickListener{
            val intent = Intent(this, StandardActivity::class.java)
            startActivity(intent)
        }
    }

项目的代码结构:
cods

运行项目
  1. 运行项目,点击按钮 standard,此时页面跳转至 StandardActivity,查看日志台:
    l1
    可以看到日志输出包含了id以及当前的Activity实例。

  2. 接着点击当前页面的按钮 standard,每跳转到新页面都这么操作,点击 3 次,查看日志台:
    l2
    可以发现,每次都创建了一个新的 StandardActivity 实例,并且由同一个管理。

  3. 点击当前页面的按钮 singleTask
    可以看到第一次点击后页面会跳转,之后再点击按钮 singleTask页面就不再跳转了。查看日志我们会发现,第一次点击按钮 singleTask会留下日志记录当前的 SingleTaskActivity实例,之后再点击按钮就不再输出这样的日志了。这个现象是由于当前Activity指定了 singleTask启动模式,当中不存在该SingleTaskActivity的实例,那么就创建一个新的SingleTaskActivity的实例;如果中存在该SingleTaskActivity的实例,那么就直接使用该实例,不再新建了。
    l3

  4. 点击当前页面的按钮 singleTop
    可以看到第一次点击后页面会跳转,之后再点击按钮 singleTop页面就不再跳转了。查看日志我们会发现,第一次点击按钮 singleTop会留下日志记录当前的 SingleTopActivity实例,之后再点击按钮就不再输出这样的日志了。
    这时再点击按钮 standard,会跳转至新页面,如果接着点击当前页面的按钮 singleTop,可以看到这一次点击后页面会跳转,之后再点击按钮 singleTop页面就不再跳转了。
    这个现象是由于SingleTopActivity指定了 singleTop启动模式,栈顶复用。
    l4

  5. 点击当前页面的按钮 singleInstance
    查看日志,可见 SingleInstanceActivity 所在的id与别的Activity不同,这是由于该Activity设定了 singleInstance启动模式,系统为其分配了一个独立的
    l5

多实验运行演示,结合对应的知识点,加深对Activity启动模式的理解。

项目代码地址

  • Java:
    https://github.com/BethelDEV/shaguaAndroid/tree/main/javaSource/ActivitySample

  • Kotlin:
    https://github.com/BethelDEV/shaguaAndroid/tree/main/kotlinSource/ActivitySample

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值