Android中Activity是由返回栈来管理的,在默认情况下,每当启动一个新的Activity,它都会在返回栈中入栈,并且出于栈的顶端。但是有些时候Activity已经在栈的顶端了,也就不需要再启动的时候重新创建一个Activity的实例了,所以我们就需要其他的启动方式。

Activity的启动方式一共分为四种:standard、singleTop、singleTask、singleInstance,可以在AndroidManiFest.xml中通过<activity>标签指定android:launchMode属性来选择启动模式。

         standard模式:该模式为Activity默认的启动模式,如果AndroidManiFest.xml中没有进行配置,系统默认使用这种方式启动,也就是每启动一个Activity,都会在返回栈的顶端插入一个新的Activity实例,我们代码来查看效果

1)新建LaunchModeDemo工程,主Activity命名为FirstActivity.java

2)FirstActivity.java

  1. package cn.lixyz.laubchmode;
  2.  
  3. import android.content.Intent;
  4. import android.support.v7.app.AppCompatActivity;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.view.View;
  8.  
  9. public class FirstActivity extends AppCompatActivity {
  10.  
  11. @Override
  12. protected void onCreate(Bundle savedInstanceState) {
  13. super.onCreate(savedInstanceState);
  14. setContentView(R.layout.activity_first);
  15.  
  16. Log.d("FirstActivity","运行了onCreate()方法创建了一个Activity实例");
  17.  
  18. findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
  19. @Override
  20. public void onClick(View v) {
  21. //从FirstActivity跳转到FirstActivity
  22. Intent intent = new Intent(FirstActivity.this,FirstActivity.class);
  23. startActivity(intent);
  24. }
  25. });
  26.  
  27. }
  28. }

3)activity_first.xml

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  3. android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
  4. android:paddingRight="@dimen/activity_horizontal_margin"
  5. android:paddingTop="@dimen/activity_vertical_margin"
  6. android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
  7.  
  8. <Button
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:text="点我跳转到另一个Activity"
  12. android:id="@+id/button"
  13. android:layout_marginTop="53dp"
  14. android:layout_alignParentTop="true"
  15. android:layout_centerHorizontal="true" />
  16. </RelativeLayout>

4)AndroidManiFest.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="cn.lixyz.laubchmode" >
  4.  
  5. <application
  6. android:allowBackup="true"
  7. android:icon="@mipmap/ic_launcher"
  8. android:label="@string/app_name"
  9. android:theme="@style/AppTheme" >
  10. <activity
  11. android:name=".FirstActivity"
  12. android:label="@string/app_name" >
  13. <intent-filter>
  14. <action android:name="android.intent.action.MAIN" />
  15.  
  16. <category android:name="android.intent.category.LAUNCHER" />
  17. </intent-filter>
  18. </activity>
  19. </application>
  20.  
  21. </manifest>

  运行结果,没点击一次按钮,都会运行一次onCreate()方法创建一个Activity的实例

  

  如果连续点击三次按钮,我们需要点击四次back才能退出到桌面,由此可见,每次点击按钮,我们都在返回栈中新创建了一个Activity放到了返回栈的顶部,我们点击back只是把最上面的Acticity退出了。

  

         singleTop模式:我们会发现,有时候明明Activity已经在栈顶端了,再创建一个实例放到顶端,这不是浪费资源么,这时候我们就需要对Activity的启动模式进行改动了,如果Activity设置为singleTop模式,那么在启动时候发现栈顶已经是该Activity的时候,系统不会在创建新的Activity实例了。

通过代码来看:

1)FirstActivity.java

  1. package cn.lixyz.laubchmode;
  2.  
  3. import android.content.Intent;
  4. import android.support.v7.app.AppCompatActivity;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.view.View;
  8.  
  9. public class FirstActivity extends AppCompatActivity {
  10.  
  11. @Override
  12. protected void onCreate(Bundle savedInstanceState) {
  13. super.onCreate(savedInstanceState);
  14. setContentView(R.layout.activity_first);
  15.  
  16. Log.d("FirstActivity","运行了onCreate()方法创建了一个Activity实例");
  17.  
  18. findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
  19. @Override
  20. public void onClick(View v) {
  21. //从FirstActivity跳转到FirstActivity
  22. Intent intent = new Intent(FirstActivity.this,FirstActivity.class);
  23. startActivity(intent);
  24. }
  25. });
  26.  
  27. }
  28. }

2)activity_first.xml

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  3. android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
  4. android:paddingRight="@dimen/activity_horizontal_margin"
  5. android:paddingTop="@dimen/activity_vertical_margin"
  6. android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
  7.  
  8. <Button
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:text="点我跳转到另一个Activity"
  12. android:id="@+id/button"
  13. android:layout_marginTop="53dp"
  14. android:layout_alignParentTop="true"
  15. android:layout_centerHorizontal="true" />
  16. </RelativeLayout>

3)AndroidManiFest.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="cn.lixyz.laubchmode" >
  4.  
  5. <application
  6. android:allowBackup="true"
  7. android:icon="@mipmap/ic_launcher"
  8. android:label="@string/app_name"
  9. android:theme="@style/AppTheme" >
  10. <activity
  11. android:name=".FirstActivity"
  12. android:label="@string/app_name"
  13. android:launchMode="singleTop">
  14. <intent-filter>
  15. <action android:name="android.intent.action.MAIN" />
  16.  
  17. <category android:name="android.intent.category.LAUNCHER" />
  18. </intent-filter>
  19. </activity>
  20. </application>
  21. </manifest>

  运行结果,只有我们启动的时候运行了一次onCreate()方法,后边点击按钮则不会再创建新的Activity实例

singleTask模式:我们使用singleTop可以使得Activity在返回栈顶端则不创建新的Activity实例,但如果Activity不在顶端,依然还会创建,如何让Activity在整个应用程序的上下文中只存在一个实例呢?这就要借助sigleTask来实现了

当一个Activity设定启动模式为singleTask的话,每次启动该活动时系统会先在整个返回栈中检查一遍,如果返回栈中已经存在该Activity的实例的话,则会直接使用该实例,并且把这个Activity之上的Activity统统出栈,如果返回栈中不存在该Activity实例,则创建一个。

通过代码来看:

1)FirstActivity.java

  1. package cn.lixyz.laubchmode;
  2.  
  3. import android.content.Intent;
  4. import android.support.v7.app.AppCompatActivity;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.view.View;
  8.  
  9. public class FirstActivity extends AppCompatActivity {
  10.  
  11. @Override
  12. protected void onCreate(Bundle savedInstanceState) {
  13. super.onCreate(savedInstanceState);
  14. setContentView(R.layout.activity_first);
  15.  
  16. Log.d("Launch","运行了FirstActivity的onCreate()方法");
  17.  
  18. findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
  19. @Override
  20. public void onClick(View v) {
  21. //从FirstActivity跳转到FirstActivity
  22. Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
  23. startActivity(intent);
  24. }
  25. });
  26.  
  27. }
  28.  
  29. @Override
  30. protected void onRestart() {
  31. super.onRestart();
  32. Log.d("Launch","运行了FirstActivity的onRestart()方法");
  33. }
  34. }

2)activity_first.xml

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  3. android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
  4. android:paddingRight="@dimen/activity_horizontal_margin"
  5. android:paddingTop="@dimen/activity_vertical_margin"
  6. android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
  7.  
  8. <Button
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:text="点我跳转到另一个Activity"
  12. android:id="@+id/button"
  13. android:layout_marginTop="53dp"
  14. android:layout_alignParentTop="true"
  15. android:layout_centerHorizontal="true" />
  16. </RelativeLayout>

3)SecondActivity.java

  1. package cn.lixyz.laubchmode;
  2.  
  3. import android.content.Intent;
  4. import android.support.v7.app.AppCompatActivity;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.view.Menu;
  8. import android.view.MenuItem;
  9. import android.view.View;
  10.  
  11. public class SecondActivity extends AppCompatActivity {
  12.  
  13. @Override
  14. protected void onCreate(Bundle savedInstanceState) {
  15. super.onCreate(savedInstanceState);
  16. setContentView(R.layout.activity_second);
  17. Log.d("Launch", "运行了SecondActivity的onCreate方法");
  18. findViewById(R.id.button2).setOnClickListener(new View.OnClickListener() {
  19. @Override
  20. public void onClick(View v) {
  21. Intent intent = new Intent(SecondActivity.this,FirstActivity.class);
  22. startActivity(intent);
  23. }
  24. });
  25. }
  26.  
  27. @Override
  28. protected void onDestroy() {
  29. super.onDestroy();
  30. Log.d("Launch", "运行了SecondActivity的onDestroy方法");
  31. }
  32. }

4)activity_second.xml

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  3. android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
  4. android:paddingRight="@dimen/activity_horizontal_margin"
  5. android:paddingTop="@dimen/activity_vertical_margin"
  6. android:paddingBottom="@dimen/activity_vertical_margin"
  7. tools:context="cn.lixyz.laubchmode.SecondActivity">
  8.  
  9. <Button
  10. android:layout_width="wrap_content"
  11. android:layout_height="wrap_content"
  12. android:text="点我跳转"
  13. android:id="@+id/button2"
  14. android:layout_alignParentTop="true"
  15. android:layout_centerHorizontal="true"
  16. android:layout_marginTop="94dp" />
  17. </RelativeLayout>

5)AndroidManiFest.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="cn.lixyz.laubchmode" >
  4.  
  5. <application
  6. android:allowBackup="true"
  7. android:icon="@mipmap/ic_launcher"
  8. android:label="@string/app_name"
  9. android:theme="@style/AppTheme" >
  10. <activity
  11. android:name=".FirstActivity"
  12. android:label="@string/app_name"
  13. android:launchMode="singleTask">
  14. <intent-filter>
  15. <action android:name="android.intent.action.MAIN" />
  16.  
  17. <category android:name="android.intent.category.LAUNCHER" />
  18. </intent-filter>
  19. </activity>
  20. <activity
  21. android:name=".SecondActivity"
  22. android:launchMode="singleTask">
  23. </activity>
  24. </application>
  25. </manifest>

  运行结果:

  程序启动,检查返回栈中没有FirstActivity的实例,调用FirstActivity的onCreate()方法创建一个实例,点击按钮跳转到SecondActivity,发现移动栈中没有SecondActivyt的实例,调用SecondActivity的onCreate()方法创建一个SecondActivity的实例,点击SecondActivity上的按钮跳转到FirstActivity上,系统发现移动栈中有FirstActivity的实例,调用FirstActivity的onRestart()方法,并且将FirstActivity实例上面的所有实例全部移除(调用SecondActivity的onDestroy()方法),再次点击FirstActivity上的按钮跳转到SecondActivity上,系统发现移动栈中没有SecondActivity的实例,于是继续调用它的onCreate()方法创建之。

singleInstance模式:指定为singleInstance模式的Activity会启用一个新的返回栈来管理这个活动。

假如我们的程序中有一个Activity是运行其他程序调用的,如果我们想实现其他程序和我们的程序可以共享这个Activity实例,上面三个启动模式是无法做到的,因为每个程序都会有自己的返回栈,而使用singleInstance模式的话,有一个单独的返回栈来管理这个Activity,不论那个程序访问这个Activity,都共用同一个返回栈。

通过代码来看:

1)FirstActivity.java

  1. package cn.lixyz.laubchmode;
  2.  
  3. import android.content.Intent;
  4. import android.support.v7.app.AppCompatActivity;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.view.View;
  8.  
  9. public class FirstActivity extends AppCompatActivity {
  10.  
  11. @Override
  12. protected void onCreate(Bundle savedInstanceState) {
  13. super.onCreate(savedInstanceState);
  14. setContentView(R.layout.activity_first);
  15.  
  16. Log.d("Launch","运行了FirstActivity的onCreate()方法,TaskId= " + getTaskId());
  17.  
  18. findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
  19. @Override
  20. public void onClick(View v) {
  21. //从FirstActivity跳转到FirstActivity
  22. Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
  23. startActivity(intent);
  24. }
  25. });
  26.  
  27. }
  28.  
  29. @Override
  30. protected void onRestart() {
  31. super.onRestart();
  32. Log.d("Launch","运行了FirstActivity的onRestart()方法,TaskId= " + getTaskId()
  33. );
  34. }
  35. }

2)activity_first.xml

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  3. android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
  4. android:paddingRight="@dimen/activity_horizontal_margin"
  5. android:paddingTop="@dimen/activity_vertical_margin"
  6. android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
  7.  
  8. <Button
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:text="点我跳转到另一个Activity"
  12. android:id="@+id/button"
  13. android:layout_marginTop="53dp"
  14. android:layout_alignParentTop="true"
  15. android:layout_centerHorizontal="true" />
  16. </RelativeLayout>

3)SecondActivity.java

  1. package cn.lixyz.laubchmode;
  2.  
  3. import android.content.Intent;
  4. import android.support.v7.app.AppCompatActivity;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.view.Menu;
  8. import android.view.MenuItem;
  9. import android.view.View;
  10.  
  11. public class SecondActivity extends AppCompatActivity {
  12.  
  13. @Override
  14. protected void onCreate(Bundle savedInstanceState) {
  15. super.onCreate(savedInstanceState);
  16. setContentView(R.layout.activity_second);
  17. Log.d("Launch", "运行了SecondActivity的onCreate方法,TaskId= " + getTaskId());
  18. findViewById(R.id.button2).setOnClickListener(new View.OnClickListener() {
  19. @Override
  20. public void onClick(View v) {
  21. Intent intent = new Intent(SecondActivity.this,FirstActivity.class);
  22. startActivity(intent);
  23. }
  24. });
  25. }
  26.  
  27. @Override
  28. protected void onDestroy() {
  29. super.onDestroy();
  30. Log.d("Launch", "运行了SecondActivity的onDestroy方法,TaskId= " + getTaskId());
  31. }
  32. }

4)activity_second.xml

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  3. android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
  4. android:paddingRight="@dimen/activity_horizontal_margin"
  5. android:paddingTop="@dimen/activity_vertical_margin"
  6. android:paddingBottom="@dimen/activity_vertical_margin"
  7. tools:context="cn.lixyz.laubchmode.SecondActivity">
  8.  
  9. <Button
  10. android:layout_width="wrap_content"
  11. android:layout_height="wrap_content"
  12. android:text="点我跳转"
  13. android:id="@+id/button2"
  14. android:layout_alignParentTop="true"
  15. android:layout_centerHorizontal="true"
  16. android:layout_marginTop="94dp" />
  17. </RelativeLayout>

5)AndroidManiFest.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="cn.lixyz.laubchmode" >
  4.  
  5. <application
  6. android:allowBackup="true"
  7. android:icon="@mipmap/ic_launcher"
  8. android:label="@string/app_name"
  9. android:theme="@style/AppTheme" >
  10. <activity
  11. android:name=".FirstActivity"
  12. android:label="@string/app_name"
  13. android:launchMode="singleTask">
  14. <intent-filter>
  15. <action android:name="android.intent.action.MAIN" />
  16.  
  17. <category android:name="android.intent.category.LAUNCHER" />
  18. </intent-filter>
  19. </activity>
  20. <activity
  21. android:name=".SecondActivity"
  22. android:launchMode="singleInstance">
  23. </activity>
  24. </application>
  25. </manifest>

  运行结果:

  发现跳转到SecondActivity的时候,TaskId变了,因为SecondActivity的启动方式是singleInstance,它启动了一个新的返回栈来管理这个Activity

Android笔记(五) Activity的启动模式的更多相关文章

  1. Android开发9——Activity的启动模式

    在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化操作.在Android中Activity的启动模式决定了Activity的启动运行方式. 一. ...

  2. Activity的启动模式全解standard,singleTop,singleTask,singleInstance

    在android中控制Activity的启动模式的属性主要控制两大功能: 1,控制activity 进入哪一个任务task 中,   有两种可能,进入启动task中,进入指定taskAffinity的 ...

  3. 无废话Android之activity的生命周期、activity的启动模式、activity横竖屏切换的生命周期、开启新的activity获取他的返回值、利用广播实现ip拨号、短信接收广播、短信监听器(6)

    1.activity的生命周期 这七个方法定义了Activity的完整生命周期.实现这些方法可以帮助我们监视其中的三个嵌套生命周期循环: (1)Activity的完整生命周期 自第一次调用onCrea ...

  4. android Activity的启动模式

    Android中Activity启动模式详解   在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化操作.在Android中Activity的启 ...

  5. Activity的启动模式(android:launchMode)

    在android里,有4种activity的启动模式,分别为: “standard” (默认) “singleTop” “singleTask” “singleInstance” 它们主要有如下不同: ...

  6. Android开发艺术2之Activity的启动模式

    Activity是Android的四大组件之一,他的重要性毋庸置疑,对于这么重要的一个组件,我们首先要知道这些都是由系统进行管理和回调的,要理解Activity的启动模式,我们首先来了解一下Andro ...

  7. 【Android - 组件】之Activity的启动模式

    Activity的启动模式目前有四种:standard.singleTop.singleTask 和 singleInstance. 1.standard standard 是标准模式,也是系统的默认 ...

  8. Android中Activity的启动模式(LaunchMode)和使用场景

    一.为什么需要启动模式在Android开发中,我们都知道,在默认的情况下,如果我们启动的是同一个Activity的话,系统会创建多个实例并把它们一一放入任务栈中.当我们点击返回(back)键,这些Ac ...

  9. Android中Activity的启动模式和使用场景

    一.为什么需要启动模式 在Android开发中,我们都知道,在默认的情况下,如果我们启动的是同一个Activity的话,系统会创建多个实例并把它们一一放入任务栈中.当我们点击返回(back)键,这些A ...

随机推荐

  1. C# WinForm程序中使用Unity3D控件 (转)

    https://www.cnblogs.com/cnxkey/articles/5394378.html 最近在自学Unity3D,打算使用这个时髦.流行.强大的游戏引擎开发一个三维业务展示系统,不过 ...

  2. 从零开始封装React UI 组件库并发布到NPM

    github 开源地址:zswui github 说明文档:wiki 1.新建目录wui (1)进入到 wui 目录 执行 npm init 命令初始化项目.更具提示信息填充将会生成的 package ...

  3. 【相机篇】从到FlyCapture2到Spinnaker

    从FlyCapture2 到 Spinnaker SDK的变换,可参见FLIR公司机器视觉的相机产品:https://www.flir.com/iis/machine-vision/ Spinnake ...

  4. 012 Android 动画效果(补间动画) +去掉App默认自带的标题+更改应用的图标

    1.介绍 补间动画开发者只需指定动画开始,以及动画结束"关键帧", 而动画变化的"中间帧"则由系统计算并补齐! 2.去掉App的标题 (1)将AndroidMa ...

  5. SQL语言(一)

    数据定义语言:简称DDL(Data Definition Language) create database 数据库名 character set 'utf-8'; drop database 数据库 ...

  6. PAT(B) 1079 延迟的回文数(Java)

    题目链接:1079 延迟的回文数 (20 point(s)) 题目描述 给定一个 k+1 位的正整数 N,写成 a​k​​⋯a​1​​a​0​​ 的形式,其中对所有 i 有 0≤a​i​​<10 ...

  7. go tcp发送网络请求

    //发送http请求 package main import ( "fmt" "net" "io" ) func main () { //使 ...

  8. Linux or Mac 重启网络

    Mac sudo ifconfig en0 down sudo ifconfig en0 up Linux /etc/init.d/networking restart

  9. go语言实现链式栈

    haa哈哈== import "errors" var ( // ErrEmpty 栈为空 ErrEmpty = errors.New("stack is empty&q ...

  10. 网易自动化测试工具(airtest)的环境部署

    airtest 环境配置: 1.安装Python2.7 及 Python3.6 版本(2个需要都安装) 2.配置python环境变量(AirtestIDE 需要在python2.x的环境下运行,所以尽 ...