Android开发艺术探索学习笔记(一)

时间:2021-08-05 04:04:03

第一章 Activity的生命周期和启动模式

  1.1Activity的生命周期全面解析

    1.1.1典型情况下的生命周期分析

    (1)在两个Activity进行切换时,当前的Activity的onPause()方法必须先执行完,新Activity的onResume()方法才会执行。

     (2)当用户打开新的Activity或者切换到桌面的时候,回调如下:onPause->onStop。这里有一种特殊情况,如果新 Activity采用了透明主题,那么当前Activity不会回调onStop,所以应该尽量避免在onPause中执行耗时的操作,onStop可以 做一些稍微重量级的回收工作,同样不能太耗时。

    1.1.2异常情况下的声明周期分析

    异常情况(比如资源相关的系统配置发生变化或者系统内存不足)

    1资源相关的系统配置发生变化(屏幕横竖屏切换)

     默认情况下,旋转手机屏幕Activity会被销毁重建,其onPause,onStop,onDestroy均会被调用,同时由于 Activity是在异常情况下终止的,系统会调用onSaveInstanceState来保存当前Activity的状态。这个方法的调用时机是在 onStop之前,它和onPause没有既定的时序关系。这个方法只会出现在Activity被异常终止的情况下,正常情况系统不会调用这个方法。当 Activity被重新创建后,系统会调用onRestoreInstanceState,并把Activity销毁时 onSaveInstanceState方法所保存的Bundle对象作为参数同时传递给onRestoreInstanceState和 onCreate方法。所以可以通过onSaveInstanceState和onRestoreInstanceState方法来判断Activity 是否被重建了。从时序上说,onRestoreInstanceState的调用时机在onStart之后。

    Activity被重 新创建后参数Bundle的接收位置可以选择onRestoreInstanceState和onCreate,二者的区别 是:onRestoreInstanceState一旦被调用,其参数savedInstanceState一定是有值的,不必额外判断时候为空;但 onCreate不行,onCreate如果是正常启动的话,其参数savedInstanceState为null,所以必须要额外判断。官方建议在 onRestoreInstanceState中进行数据的恢复。

    再次注意一点:系统只在Activity异常终止的时候才会调用onSaveInstanceState和onRestoreInstanceState方法来存储和恢复数据,其他情况不会触发这个过程。

    2内存不足导致低优先级的Activity被杀死

    Activity的优先级由高到低分三种:

    1前台Activity;

    2可见但非前台(比如Activity中弹出了一个对话框);

    3后台Activity;

    如果一个进程没有四大组件在运行,那么它将会很快被杀死,所以,最好是将后台工作放在service中保证进程有一定的优先级,这样就不会轻易的被系统杀死。

  如果不想让Activity在屏幕旋转时重新创建,可以给该Activity添加如下属性:

//4.0以下版本
android:configChanges=“orientation”
//4.0以上版本
android:configChanges=“orientation|screenSize”

  

  1.2Activity的启动模式

    1.2.1 Activity的LaunchMode    

    Activity的四种启动模式:standard,singleTop,singTask,singInstance。

    (1)standard—标准模式

    系统默认的模式,每次启动一个Activity都会重新创建一个新的实例,不管这个实例是否已经存在。(注意使用ApplicationContext启动Activity会报AndroidRuntimeException(比如在广播接收器中),这是因为Context没有任务栈无法让Activity加入所致,解决办法是在Activity启动时指定FLAG_ACTIVITY_NEW_TASK标记位,这个时候Activity是以singTask模式启动的)。

    (2)singTop—栈顶复用模式

    如果新Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时会触发onNewIntent方法。

    (3)singleTask—栈内复用模式

    只要Activity在栈内存在,那么多次启动此Activity都不会创建新的Activity实例,onNewIntent方法同样会被触发。(注意该模式具有cleartop的效果,会导致栈内所有在该Activity上面的Activity全部出栈,例如ABCD代表一个任务栈S内的四个Activity,A位于栈底D位于栈顶,如果B为singleTask模式,那么重新启动B后栈内会变为AB,CD会被移出栈)

    (4)singleInstance—单实例模式

    一种加强的singleTask模式,具有此种模式的Activity只能单独的位于一个任务栈中,后续的请求均不会创建新的Activity,除非这个独特的任务栈被系统销毁。

  Activity指定启动模式的两种方法:
    (1)AndroidManifest.xml制定(P20文件写错了)    

       <activity
android:name=".MainActivity"
android:configChanges="orientation|screenSize"
android:label="@string/app_name"
android:launchMode="singleTop"
android:theme="@style/AppTheme.NoActionBar">

    (2)Intent设置标志位    

Intent intent=new Intent();
intent.setClass(MainActivity.this,SecondActivity.class);
intent.setFlags(Inent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);

    注意第二种的优先级要高于第一种,当两种同时存在时,以第二种方式为准;

    1.2.2 Activity的Flags

    几个常用的标记位    

    (1)FLAG_ACTIVITY_NEW_TASK  为Activity指定singleTask启动模式。

    (2)FLAG_ACTIVITY_SINGLE_TOP  为Activity指定singleTop启动模式。

    (3)FLAG_ACTIVITY_CLEAR_TOP  具有此标记位的Activity当它启动时,在同一个任务栈中所有位于它上面的Activity都要出栈。

    (4)FlAG_ACTIVITY_EXCLUDE_FROM_RECENTS 具有这个标记的Activity不会出现在历史Activity列表中,等同于android:excludeFromRecents="true";

  1.3IntentFilter的匹配规则

    IntentFilter中的过滤信息有三种,分别是action,category,data,一个IntentFilter中action,category,data可以有多个,一个Activity中可以有多个IntentFilter,一个Intent只要能匹配任何一组IntnetFilter就可以成功启动。

    1.action的匹配规则

    Intent中的action存在且必须和过滤规则中的其中一个action相同,这里说的匹配是指action中的字符串值完全一样,所以action是区分大小写的。

    2.category的匹配规则

    Intent中只要有category,不管有几个,每个都要能够和过滤规则中的任何一个category相同,Intent中不设置category也可以匹配。

    3.data的匹配规则

    和action类似,过滤规则中定义了data,Intent中也要定义可匹配的data。

  Intent-filter的匹配规则对于Service和BroadcastReceiver也是同样的道理,不过系统对于service的建议是尽量使用显式调用方式来启动服务。