需求描述

在应用Application1中存在A、B两个activity,当在应用启动了A、B activity,点击Recent键,如何让A、B两个activity都显示在Recent界面(最近任务)?


需求分析

根据Android SDK介绍,Recent中是管理最近的任务(Task);熟悉Activity的可以知道Activity是隶属于某个Task的,不熟悉Activity的可以阅读下面的帖子:

http://blog.csdn.net/ff20081528/article/details/17219951#comments

那么了解了Activity的一些知识后,我们就可以推断出要让A、B在Recent中显示出来,必须将A、B处于不同的Task中;那么需要设置A、B的launchMode为singleInstance;

跟”singleTask”一样.除了系统不能再启动其它activity到拥有这个activity实例的任务中.activity永远是任务的唯一;任何由这个activity启动的其它activity都在另一个任务中打开.也就可以说singleInstance必然会开辟出一个新的Task。


Demo示例

MainActivity.java:

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

    Button btn1;

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); btn1 = (Button) findViewById(R.id.button);
btn1.setOnClickListener(this); } @Override
public void onClick(View view) {
switch (view.getId()){
case R.id.button:
Intent intent1 = new Intent(this,ActivityA.class);
intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent1);
break;
}
}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25

ActivityA,java

public class ActivityA extends AppCompatActivity {

    Button btn2;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_activity);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar); FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
}
}); btn2 = (Button) findViewById(R.id.button2);
btn2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent2;
intent2 = new Intent(ActivityA.this,ActivityB.class);
startActivity(intent2);
}
});
} }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31

Manifest.xml

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="lanchmode.sunrise.com.lanchmode"> <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=".ActivityA"
android:label="@string/title_activity_activity"
android:launchMode="singleInstance"
android:theme="@style/AppTheme.NoActionBar" />
<activity
android:name=".ActivityB"
android:label="@string/title_activity_b"
android:launchMode="singleInstance"
android:theme="@style/AppTheme.NoActionBar" />
</application> </manifest>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30

运行程序,然后使用adb shell dumpsys activity activities命令查看activity的栈的信息:

Running activities (most recent first):
TaskRecord{b1c230d #265 A=lanchmode.sunrise.com.lanchmode U=0 StackId=1 sz=1}
Run #3: ActivityRecord{638abae u0 lanchmode.sunrise.com.lanchmode/.ActivityB t265}
TaskRecord{7b648d3 #264 A=lanchmode.sunrise.com.lanchmode U=0 StackId=1 sz=1}
Run #2: ActivityRecord{65e4f8b u0 lanchmode.sunrise.com.lanchmode/.ActivityA t264}
TaskRecord{8ff7c10 #263 A=lanchmode.sunrise.com.lanchmode U=0 StackId=1 sz=1}
Run #1: ActivityRecord{5a771a1 u0 lanchmode.sunrise.com.lanchmode/.MainActivity t263}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

可以看出A、B及主activity都处于不同的Task中,那么我们点击Recent看下,是否已经实现了我们需求了呢: 

结果很是失望,只有最后启动的B在最近任务中。


需求出现问题解决

不同的Activity已经处于不同的Task了,为什么还是在最近任务中没有分别显示呢?是因为在同一个应用中(但是Recent的描述就是管理最近任务,任务就是Task啊)?通过查看activity的属性,发现了taskAffinity这个属性;

  • taskAffinity表示当前activity具有亲和力的一个任务(翻译不是很准确,原句为The task that the activity has an affinity for.),大致可以这样理解,这个 taskAffinity表示一个任务,这个任务就是当前activity所在的任务。
  • 在概念上,具有相同的affinity的activity(即设置了相同taskAffinity属性的activity)属于同一个任务。

  • 一个任务的affinity决定于这个任务的根activity(root activity)的taskAffinity。

  • 这个属性决定两件事:当activity被re-parent时,它可以被re-paren哪个任务中;当activity以FLAG_ACTIVITY_NEW_TASK标志启动时,它会被启动到哪个任务中。(这个比较 难以理解,请结合中的属性allowTaskReparenting和Intent中的标志 FLAG_ACTIVITY_NEW_TASK加以理解)

  • 默认情况下,一个应用中的所有activity具有相同的taskAffinity,即应用程序的包名。我们可以通过设置不同的taskAffinity属性给应用中的activity分组,也可以把不同的 应用中的activity的taskAffinity设置成相同的值。

  • 为一个activity的taskAffinity设置一个空字符串,表明这个activity不属于任何task。

那么就是就使用taskAffinity设置A、B的亲和力:

        <activity
android:name=".ActivityA"
android:label="@string/title_activity_activity"
android:launchMode="singleInstance"
android:taskAffinity="com.sunrise.A"
android:theme="@style/AppTheme.NoActionBar" />
<activity
android:name=".ActivityB"
android:label="@string/title_activity_b"
android:taskAffinity="com.sunrise.B"
android:launchMode="singleInstance"
android:theme="@style/AppTheme.NoActionBar" />
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

然后再运行程序,首先查看栈情况:

Running activities (most recent first): 
TaskRecord{b1c230d #265 A=lanchmode.sunrise.com.lanchmode U=0 StackId=1 sz=1} 
Run #3: ActivityRecord{638abae u0 lanchmode.sunrise.com.lanchmode/.ActivityB t265} 
TaskRecord{7b648d3 #264 A=lanchmode.sunrise.com.lanchmode U=0 StackId=1 sz=1} 
Run #2: ActivityRecord{65e4f8b u0 lanchmode.sunrise.com.lanchmode/.ActivityA t264} 
TaskRecord{8ff7c10 #263 A=lanchmode.sunrise.com.lanchmode U=0 StackId=1 sz=1} 
Run #1: ActivityRecord{5a771a1 u0 lanchmode.sunrise.com.lanchmode/.MainActivity t263}

栈的情况没有变化,然后再点击Recent,运行如下图 

结果是OK了!!!


总结

如果需要将同一个APP的不同Activity在Recent中显示,需满足如下两点:

    1. LaunchMode设置为singleInstance。
    2. 设置android:taskAffinity参数。

如何将同一个APP中的不同activity在Recent(最近任务)中显示?的更多相关文章

  1. Android中在不同activity中进行自定义广播的解析

    相信有不少人和我一样曾经尝试过在同一个项目中的两个activity进行广播,发现怎么都实现不了.我也是困惑了好久才发现,这么搞本来就是不行的.首先在同一个项目下不同的activity之间广播没有意义, ...

  2. Android中突发情况Activity数据的保存和恢复

    Android中突发情况Activity数据的保存和恢复 写在前面:在我们的APP使用的过程中,总有可能出现各种手滑.被压在后台.甚至突然被杀死的情况.所以对APP中一些临时数据或关键持久型数据,就需 ...

  3. ActionBar官方教程(6)把图标变成一个返回到上级的按钮,同一个app间,不同app间,不同fragment间

    Navigating Up with the App Icon Enabling the app icon as an Up button allows the user to navigate yo ...

  4. android开发中关于继承activity类中方法的调用

    android开发中关于继承activity类中的函数,不能在其他类中调用其方法. MainActivity.java package com.example.testmain; import and ...

  5. 在Activity,Service,Window中监听Home键和返回键的一些思考,如何把事件传递出来的做法!

    在Activity,Service,Window中监听Home键和返回键的一些思考,如何把事件传递出来的做法! 其实像按键的监听,我相信很多人都很熟练了,我肯定也不会说这些基础的东西,所以,前期,还是 ...

  6. Appium+Python自动化 3 -获取 app 包名和 activity

    方法一: ①手机通过USB连接电脑 ②打开手机上被测app ③在电脑上 dos命令窗口,输入命令 adb shell dumpsys window w | findstr \/ | findstr n ...

  7. 如何判断是否为同一个App,Ionic3如何修改包名

    如何判断是否同一个App 使用Ionic3创建了两个项目demo1.demo2,然后使用同一个JDK,生成了两个不同的keystore证书. 结果在手机端安装的时候,先安装demo1,没有任何替换的提 ...

  8. Django 多数据库联用(同一个APP的models里不同class用不同数据库)

    很多网站有多数据库联用的文章,如自强学堂http://code.ziqiangxuetang.com/django/django-multi-database.html 大都只讲解如何让不同的app对 ...

  9. Android 程序中得到root activity的引用

    今天写anroid时,想获得一个root activity的引用. ios中这个很简单, [UIApplication sharedApplication].keyWindow 得到window对象, ...

随机推荐

  1. 一本通1635【例 5】Strange Way to Express Integers

    1635:[例 5]Strange Way to Express Integers sol:貌似就是曹冲养猪的加强版,初看感觉非常没有思路,经过一番艰辛的***,得到以下的结果 随便解释下给以后的自己 ...

  2. Spring定时服务QuartZ

    在JavaEE系统中,我们会经常用到定时任务,比如每天凌晨生成前天报表,每一小时生成汇总数据等等. 我们可以使用java.util.Timer结合java.util.TimerTask来完成这项工作, ...

  3. 【题解】 bzoj2435: [Noi2011]道路修建 (傻逼题)

    bzoj2435,懒得复制,戳我戳我 Solution: 模拟即可(有点傻逼啊 Code: //It is coded by Ning_Mew on 5.13 #include<bits/std ...

  4. MySQL服务器最大连接数的合理设置

    MySQL服务器的连接数并不是要达到最大的100%为好,还是要具体问题具体分析,下面就对MySQL服务器最大连接数的合理设置进行了详尽的分析,供您参考. 我们经常会遇见“MySQL: ERROR 10 ...

  5. 滚动条事件,当页面滚动到距顶部一定高度时某DIV自动隐藏和显示

    $(function () {                        //绑定滚动条事件              //绑定滚动条事件            $(window).bind(&q ...

  6. KVM安装、镜像创建(一)

    环境准备 VMware Workstation Pro启动虚拟化 查看启动的系统是否支持vmx或svm grep -E '(vmx|svm)' /proc/cpuinfo 备注:操作系统centos ...

  7. 搜索引擎:Elasticsearch与Solr

    搜索引擎选型调研文档 Elasticsearch简介* Elasticsearch是一个实时的分布式搜索和分析引擎.它可以帮助你用前所未有的速度去处理大规模数据. 它可以用于全文搜索,结构化搜索以及分 ...

  8. 流媒体技术学习笔记之(四)解决问题video.js 播放m3u8格式的文件,根据官方的文档添加videojs-contrib-hls也不行的原因解决了

    源码地址:https://github.com/Tinywan/PHP_Experience 总结: 说明: 测试环境:本测试全部来自阿里云直播和OSS存储点播以及本地服务器直播和点播 播放器:Vid ...

  9. Javascript非构造函数的继承

    一.什么是"非构造函数"的继承? 比如,现在有一个对象,叫做"中国人". var Chinese = { nation:'中国' }; 还有一个对象,叫做&qu ...

  10. [原]Android Fragment 入门介绍

    Fragment Fragment 产生,优点,用途,使用方法简介 1 Fragmeng简介 自从Android 3.0中引入fragments 的概念,根据词海的翻译可以译为:碎片.片段.其上的是为 ...