声明:作者原创,转载注明出处。
作者:帅气陈吃苹果
一、Activity窗口结构
Activity是Android四大组件之一。Activity窗口结构由Activity、ViewRoot、WindowManager和WindowManagerService构成,四者的关系如下图:
在应用程序中,通常情况下,一个Activity对应一个单独的屏幕,但Activity不负责视图控制,而是负责生命周期的控制和事件的处理,真正控制视图的是Window。一个Activity包含一个Window,Window才真正代表一个窗口。Activity就像一个控制器,统筹视图的添加与显示,以及通过其他回调方法,来与Window、View组件进行交互。
Window中持有一个DectorView,DectorView是FrameLayout的子类,它是我们自定义布局元素的根布局,可以被认为是Android视图树的根节点视图。DectorView内部包含一个垂直走向的LinearLayout,这个LinearLayout根据不同Android版本有不同的布局结构,通常情况下,其被分成两个部分,上面的部分是标题栏(如上图TitleView
),下面的部分是内容栏(如上图ContentViews
),内容栏的id是content
,我们在Activity中通过setContentView()所设置的自定义布局内容其实是被添加到内容栏部分的。我们可以通过下列方式来得到布局文件的根布局:
ViewGroup content = (ViewGroup) findViewById(android.R.id.content);
ViewGroup rootView = (ViewGroup) content.getChildAt(0);
也就是说,可以通过先得到DectorView中的content
部分,再得到这个内容栏的第一个元素,当然,content
也只能有一个直接子元素。
View的绘制和事件分发都是通过ViewRoot来执行或传递的。ViewRoot是连接WindowManagerService和DectorView的纽带,View的三大流程:测量(Measure)、布局(Layout)和绘制(Draw)都是通过ViewRoot来完成的。ViewRoot实现了ViewParent接口,继承了Handler类,可以接收和分发事件,Android中所有的触屏事件、按键事件、界面刷新等事件都是通过ViewRoot来进行分发的。
参考链接:简析Window、Activity、DecorView以及ViewRoot之间的错综关系---作者:Ruheng
更多Activity窗口机制知识,详见Android窗口机制系列---作者:Hohohong
二、Activity创建和注册
创建:新建一个类继承Activity类或其子类,实现必要的生命周期回调方法,如onCreate()
;
注册:在AndroidManifest.xml
中进行注册,如下:
<activity android:name=".MainActivity"></activity>
其中name
属性中需要包含Activity所在包的包名及Activity类名,若已在<Manifest>
标签中声明当前默认包,则在Activity类名前加一个点“.”即可。
若注册的Activity为启动Activity,则需在<activity>
标签中添加下列代码,表示该Activity为应用程序启动后用户见到的第一个Activity,启动Activity有且只有一个:
<intent-filter>
<action android:name="android.intent.action.Main" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
三、Activity生命周期
创建:
09-18 12:04:51.357 29452-29452/com.sqchen.androidtest D/MAIN_TAG: onCreate
09-18 12:04:51.359 29452-29452/com.sqchen.androidtest D/MAIN_TAG: onStart
09-18 12:04:51.361 29452-29452/com.sqchen.androidtest D/MAIN_TAG: onResume
销毁:
09-18 12:04:54.260 29452-29452/com.sqchen.androidtest D/MAIN_TAG: onPause
09-18 12:04:54.717 29452-29452/com.sqchen.androidtest D/MAIN_TAG: onStop
09-18 12:04:54.717 29452-29452/com.sqchen.androidtest D/MAIN_TAG: onDestroy
运行中--->按下HOME键/被新Activity覆盖/锁屏:
09-18 13:02:41.389 18317-18317/com.sqchen.androidtest D/MAIN_TAG: onResume
09-18 13:02:44.924 18317-18317/com.sqchen.androidtest D/MAIN_TAG: onPause
09-18 13:02:44.961 18317-18317/com.sqchen.androidtest D/MAIN_TAG: onStop
从HOME或Activity返回/解屏:
09-18 13:07:38.467 22786-22786/com.sqchen.androidtest D/MAIN_TAG: onStart
09-18 13:07:38.468 22786-22786/com.sqchen.androidtest D/MAIN_TAG: onResume
旋转屏幕导致Activity重建:
09-18 13:50:26.562 26975-26975/com.sqchen.androidtest D/MAIN_TAG: onPause
09-18 13:50:26.563 26975-26975/com.sqchen.androidtest D/MAIN_TAG: onStop
09-18 13:50:26.563 26975-26975/com.sqchen.androidtest D/MAIN_TAG: onDestroy
09-18 13:50:26.619 26975-26975/com.sqchen.androidtest D/MAIN_TAG: onCreate
09-18 13:50:26.621 26975-26975/com.sqchen.androidtest D/MAIN_TAG: onStart
09-18 13:50:26.624 26975-26975/com.sqchen.androidtest D/MAIN_TAG: onResume
onRestart
:表示Activity正在重新启动,当Activity从不可见重新变成可见状态时,触发该方法;
onPause
:表示Activity正在停止,此时可以做一些存储数据、停止动画等工作,但是不能太耗,因为会影响新Activity显示,旧Activity的onPause
先执行,然后才执行新Activity的onResume
。
注意:当Activity中弹出Dialog时,不会触发onPause,然而当Activity启动Dialog风格的Activity时,会回调onPause方法。
参考链接:Android四大组件---作者:Lemon_95
四、Activity数据恢复
Activity的数据恢复可以借助于onSaveInstanceState()
方法来完成,该方法用于完成对一些临时的、非永久性的数据的存储,这样的临时数据包括:EditText中已经输入的内容、CheckBox选中状态、ScrollView的滑动位置、当前视频的播放位置等。
当Activity异常销毁时,就会触发onSaveInstanceState()
方法。什么情况属于异常销毁呢?一般认为,除了用户主动按下Back键的行为外,导致Activity被销毁的情况,都属于异常销毁,如:
- 点击锁屏键;
- 点击HOME键;
- 其他APP进入前台;
- 启动了另一Activity;
- 屏幕方向改变(垂直或水平的改变);
- APP所在进程被杀死;
注意:onSaveInstanceState()
方法先于onStop
方法调用,但与onPause
无必然先后顺序。
数据保存
在onSaveInstanceState
保存数据:
@Override
protected void onSaveInstanceState(Bundle outState) {
Debug.i("保存数据");
}
数据恢复
在onRestoreInstanceState
恢复数据:
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
Debug.i("恢复数据");
}
一般情况下我们在onCreate
方法里进行数据的恢复处理,不过需要注意判断Bundle对象参数是否为null:
@Override
protected void onCreate(Bundle savedInstanceState) {
setContentView(R.layout.activity_main);
if(savedInstanceState != null) {
//数据恢复处理
}
}
数据持久化
API 21中为Activity增加了一个android:persistableMode
属性,其值如下:
persistNever //从不,不调用那两个方法持久化页面的数据和状态
persistRootOnly //默认模式,仅仅会作用在根Activity或者Task中
persistAcrossReboots //重启设备,会持久化页面的数据和状态
只要在AndroidManifest.xml
将对应Activity的persistableMode
属性设置为persistAcrossReboots
,再实现下列回调方法,即可让Activity实现数据持久化。
@Override
public void onCreate(@Nullable Bundle savedInstanceState, @Nullable PersistableBundle persistentState) {
super.onCreate(savedInstanceState, persistentState);
//从PersistableBundle中取出保存的数据,进行恢复
}
@Override
public void onRestoreInstanceState(Bundle savedInstanceState, PersistableBundle persistentState) {
Debug.i("持久化恢复数据");
}
@Override
public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState) {
Debug.i("持久化保存数据");
}
onSaveInstanceState
方法和onRestoreInstanceState
方法不在“正常”的生命周期中,只在一些突发异常情况下才会触发,比如横屏切换、按下HOME键等,而API 21中在Activity的这几个方法中增加了PersistableBundle
参数,使得这些方法有了系统关机重启后数据恢复的能力。
参考链接: