Activity&&Intent


title: Activity&&Intent
tags:安卓基础知识
grammar_cjkRuby: true


Activity的状态

...

Activity的生命周期

Activity&&Intent_第1张图片
activity_lifecycle.png

Activity的正常生命周期

onCreate():Activity正在创建;
onStart():Activity可见但不能交互;
onResume():Activity可见可交互;
onPause():Activity可见不可交互,在此方法中中不能做耗时操作,会影响新Activity现实.因为onPause执行完,才会执行新Activity的onResume;
onStop():Activity不可见不可交互,同样不能太耗时;
onDestory():Activity即将销毁,在这里可以做回收工作和最终资源的释放;
onRestart():Activity正在重启.一般,当当前Activity从不可见重新变为可见时,onRestart就会调用.这种情况通常是用户按Home键切换到桌面或者用户打开了一个新的Activity,当前的Activity执行onPause和onStop,然后又回到这个Activity,就会出现这种情况;

流程示例:ActivityA-->ActivityB-->ActivityA
onCreate(A)-->onStart(A)-->onResume(A);A创建
onPause(A)-->onCreate(B)-->onStart(B)-->onResume(B)-->onStop(A);
A跳转B,A先变为可见不可交互,然后创建B,B创建完成后并可见可交互之后,A彻底不可见不可交互;
onPause(B)-->onStart(A)-->onResume(A)-->onStop(B)-->onDestroy(B);
关闭B返回A,同上,B先变为可见但不可交互,A因为已经创建过,所以只执行可见可交互,然后B彻底不可见不可交互并销毁;

Activity的异常生命周期

默认情况下,Activity被销毁并重新创建,生命周期如下:
onSaveInstanceState-->onStop-->onDestory;异常销毁,并保存状态
onCreate-->onStart-->onRestoreInstanceState;重新创建,传递状态

安卓系统保存恢复异常停止Activity状态的流程

当Activity异常销毁时,系统自动帮我们做了一些保存并恢复Activity状态的操作;
首先Activity被异常终止时,Activity调用onSaveInstanceState保存数据,Activity会委托Window去保存数据,接着Window再委托它上面的顶级容器去保存数据.顶级容器是一个ViewGroup,一般来说很可能是DecorView.最后顶级容器再去一一通知它的子元素来保存数据,通过每个子元素内的onSaveInstanceState和onRestoreInstanceState的各自实现来保存Activity的状态;

Activity的异常生命周期情况

资源相关的系统配置改变导致Activity被杀死并重新创建

通过设置configChanges属性,可以让Activity在配置改变时不重新创建;

Activity&&Intent_第2张图片
安卓系统配置属性.png

上图中比较常用的只有locale、orientation、keyboardHidden这三个选项;
想要配置改变不重新创建,如下设置:


并且系统会调用onConfigurationChanged方法,在方法中可以根据需要特殊处理;

资源内存不足导致低优先级的Activity被杀死并重新创建

Activity的优先级(高-->低)
  • 前台Activity
  • 可见不可交互Activity-->比如Activity中弹出对话框
  • 不可见不可交互Activity(后台Activity)-->已经被暂停的activity,比如执行了onStop的

提示:onCreate和onRestoreInstanceState恢复数据的区别
onRestoreInstanceState被调用时,其参数Bundle必然是有数据的;
onCreate获取被保存的数据,需要判断Activity正常启动的话Bundle为空的情况;

启动模式

standard

多个ActivityA

标准模式,也是系统的默认模式.每次启动一个Activity都会创建一个新的实例,不管这个实例是否存在.在这种模式下,Activity默认会进入启动它的Activity所在的任务栈中.比如A启动(跳转)B,B就会在A的任务栈中.这就是为什么传入ApplicationContext启动Activity时会报错的原因,因为非Activity类型的Context;解决这个问题的办法,可以在启动时添加个标记位FLAG_ACTIVITY_NEW_TASK,也就是以singleTask的模式启动的.

singleTop

仅位于栈顶时一个栈一个ActivityA

栈顶单一模式.在这种模式下,如果新Activity已经存在于任务栈的栈顶,Activity不会重新创建,同时调用onNewIntent方法,注意不会调用onCreate和onStart方法;非栈顶情况,仍然创建;

eg:应用场景举例
比如从通知栏点击跳转APP的Activity,每次都会创建个新实例,使用singleTop模式后,每次跳转的都是同一实例;

singleTask

一个栈一个ActivityA

栈内单一模式.在这种模式下,只要Activity在栈内存在,多次启动Activity都不会重新创建,并且也会调用onNewIntent方法.具体的说ActivityA,系统首先寻找是否存在任务栈,如果不存在,就创建个新的任务栈,并把A压入栈中.如果存在任务栈,判断栈中有无A的实例,有实例的情况,会把Activity调到栈顶并调用onNewIntent方法;没有实例的情况,创建A的实例并压入栈中.

注意:
1.singleTop具有clearTop效果,在singleTask中把ActivityA调到栈顶时会导致原Activity上的所有Activity出栈.

2.singleInstance或singleTask和onActivityResult回调失效?
当ActivityA启动ActivityB,且A和B都为singleInstance或singleTask的情况下,onActivityResult回调会失效?猜测是因为两个任务栈的原因...记不清了

singleInstance

一个App一个ActivityA

单一实例模式.在这种模式下,整个应用中只有一个Activity实例.这种模式的Activity只能单独位于单一的任务栈中.比如当启动ActivityA时,系统会为它创建一个单一的任务栈,栈中只会存在Activity.当再次启动ActivityA时,会从其他栈切换到Activity所在的任务栈并复用ActivityA;

使用启动模式

两种方式都可以为Activity指定启动模式,但还是有区别的.
1.在代码中使用时优先级要高于mainfest中使用;
2.在mainfest中使用时无法为Activity指定FLAG_ACTIVITY_CLEAR_TOP等标记,在代码中使用时无法指定singleInstance模式;

mainfest中使用


        

代码中使用

 Intent intent=new Intent(this,Demo1Activity.class);
 intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);

任务栈

在安卓系统中,使用后进先出的栈结构保存Activity.默认情况下,所有Activity的任务栈名字为应用包名.也可以通过设置askAffinity属性来指定Activity的任务栈.

任务栈分为前台任务栈和后台任务栈,后台任务栈中的Activity处于暂停状态,用户可以通过切换将后台任务栈再次调到前台.


进程优先级

Activity的Flags

Activity的标记位有很多,这里主要介绍些常用的,其他可以查看官方文档:

  • FLAG_ACTIVITY_NEW_TASK singleTask启动模式
  • FLAG_ACTIVITY_SINGLE_TOP singleTop启动模式
  • FLAG_ACTIVITY_CLEAR_TOP 通常和FLAG_ACTIVITY_SINGLE_TOP一起用,清除到栈顶的所有Activity
  • FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS 具有这个标记的Activity不会出现在历史Activity列表中,当某些情况下我们不希望用户通过历史列表回到我们的Activity的时候这个标记比较有用.在xml中也可以指定此属性

IntentFilter的匹配规则

Activity启动分为显式调用和隐式调用:

显式调用:需要明确被指定对象的组件信息,包括包名和类名;
隐式调用:需要匹配IntentFilter的所有配置信息;
同一个Activity可能存在多个IntentFilter,一个Intent只要能匹配任意一组就可以成功启动Actvitiy

action的匹配规则

action是一个字符串,系统预定义了一些action,我们也可以在应用中定义自己的action.
action的匹配要求是Intent中的action存在且必须和过滤规则中的其中一个action相同.
action区分大小写.

category的匹配规则

category的匹配规则与action不同,每个category都必须能匹配过滤规则中已定义的category.
Intent也可以没有category,仍然可以匹配成功.因为系统在启动Actviity时默认会为Intent加上"android.intent.category.DEFAULT".为了我们的Activity能够接收隐式调用,必须在intent-filter中指定这个category.

data的匹配规则

组成结构

data与action的匹配规则类似,如果过滤规则中定义了data,Intent中必须有一个data能与之匹配.data由mimeType和URI两个部分组成.mimeTyp指媒体类型,比如image/jpeg、video/*等,可以表示图片、文本、视频等不同的媒体格式.下面是URI的结构:

://:/[||].

eg:
cotent://com.example.project:200/folder/subfolder/etc
http://www.baidu.com:80//search/info

scheme:模式.如果没有指定scheme,那么整个URI无效.
host:主机名.同上
port:端口号.仅当URI指定了scheme和host的情况下,port才有意义.
其余三个参数:表示路径信息.path表示完整的路径信息,pathPrefix表示可包含通配符的完整路径信息,pathPattern表示路径的前缀信息.

匹配规则

对于没有指定明确URI的,如下:


                
            

这种规则指定了媒体类型为所有类型的图片,即Intent中的mimeType属性同样为"image/*"才能匹配,这种情况过滤规则虽然没有指定URI,但是却有默认值,URI的默认值为content和file.

对于明确指定了mimeType和URI的,按照匹配规则匹配就行.如下


                
            
            
intent.setDataAndType(Uri.parse("http://abc)"),"video/mpeg");

Intent-filter的匹配规则对于Service和BroadcastReciver也是一样的.最后在实际开发中,隐式启动Activity时,需要做下判断,是否有Activity能够匹配到我们隐式的Intent,通过PackageManager或者Intent的resolveActivity方法,如果匹配不到会返回null.

注意:
如果要为Intent指定完整的data,不能先调用setData再调用setType,因为setData和setDataAndType两个方法会彼此清除掉对方的值.

Type的匹配规则

...

component的匹配规则

...

extras的匹配规则

...

Flag的匹配规则

...

你可能感兴趣的:(Activity&&Intent)