Android基础复习——Activity
Activty 是什么
Activity 是 Android 四大应用组件(分别是Activity,Service,Content Provider, Broadcast Receiver)之一。
Activity 为用户提供在 Android 设备屏幕上交互的界面。不管用户是打电话,发短信,刷微博,都会有 Activity 的身影。
不信的话,我们可以使用 adb 命令查看当前界面的 Activity:
adb shell dumpsys window windows | grep -E 'mCurrentFocus|mFocusedApp'
通过上面的结果,我们可以知道酷安APP当前界面的Activity是com.coolapk.market/.view.main.MainActivity
Activty 怎么用
Activity 的生命周期
- 创建我们自己的Activty,一般是继承自
Activity
或是它的子类(如AppCompactActivity
)。 - 在 Activity 的生命周期方法中,实现我们自定义界面或业务逻辑。在
onCreate()
方法中调用setContentView()
设置 Activity
的界面。 - 像
onCreate()
方法一样,Activity
中有类似这样一共七个生命周期方法。
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
}
override fun onStart() {
super.onStart()
}
override fun onResume() {
super.onResume()
}
override fun onPause() {
super.onPause()
}
override fun onStop() {
super.onStop()
}
override fun onDestroy() {
super.onDestroy()
}
override fun onRestart() {
super.onRestart()
}
}
那么问题来了,不同的生命周期方法,我们对应应该做哪些事情呢。
在这之前,我们必须了解Activity的三种状态。
- 运行中(resumed): Activity 位于屏幕前台并且有焦点
- 暂停(paused): Activity 可见(但是有其他Activity处于运行中的状态)
- 停止(stopped): Activity处于后台,被其他Activity完全遮挡
方法 | 说明 | 做什么 |
---|---|---|
onCreate | 在创建Activity时调用 | 创建视图,绑定数据等 |
onStart | Activity即将可见 | 绘制元素,运行动画 |
onResume | Activity开始与用户交互,Activity位于Activity栈顶层且有焦点 | |
onPause | 系统开始启动其他Activity时调用 | 确认对持久性数据的未保存更改、停止动画以及其他可能消耗 CPU 的内容 ps:不能执行耗时的操作,否则其他Activity在它执行完之前不能执行 |
onStop | 对用户不可见时调用 | 停止UI刷新,运行动画以及绘制元素 |
onDestroy | 在Activity被销毁时调用 | 执行Activity销毁前的清除工作 |
onRestart | 在 Activity 已停止并即将再次启动前调用。 |
下面这个图时Android生命周期流程图,很好的描述了Activity的完整生命周期以及其中的钩子方法。
其中onCreate到onDestroy是Activity的完整生命周期
onStart和onStop是可见生命周期
onResume和onPause是前台生命周期
Activity 的注册
在AndroidManifest文件中声明注册Activity,在android:name
中声明要注册的Activity
的类名。其他属性可以参考文档
<?xml version="1.0" encoding="utf-8"?>
<manifest ...>
<application ...>
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>
</manifest>
intent-filter
是为应用组件声明的Intent过滤器,通过这些过滤器(根据 Intent 的操作、数据和类别指定自身接受的 Intent 类型)过滤被传递到应用组件(如Activity)的隐式Intent。
Activity 的启动
- 启动Activity( startActivity方法)
val intent = Intent(this,SecondActivity::class.java)
intent.putExtra("data", 123)//传递参数
startActivity(intent)
- 启动Activity并获得结果 ( startActivityForResult 方法)
val pickIntent = Intent(Intent.ACTION_PICK, ContactsContract.Contacts.CONTENT_URI)
startActivityForResult(pickIntent, PICK_CONTACT_REQUEST)
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
if(resultCode == Activity.RESULT_OK && requestCode == PICK_CONTACT_REQUEST) {
//处理返回结果
}
Activity的启动模式
共有四种启动模式:standard,singleTop,singleTask,singleInstance
启动模式 | 特点 | 应用场景 |
---|---|---|
standard | 启动此模式的Activity,会创建一个新的Activity的实例 | 默认场景,通常Activity都是使用这种 |
singleTop | 分两种情况:1.启动的Activity在栈顶,不创建新的实例,调用Activity的onNewIntent方法 2.不在栈顶,创建新的实例 | 适合有就更新,没有就创建的场景(如搜索) |
singleTask | 只有一个实例,若不存在则创建,反之,此Activity之上的其他Activity 都出栈 | APP的入口,当通过其他方式调用APP的时候,不会反复创建主页面。例如其他APP调用自己的应用的时候。 |
singleInstance | 在新的任务(Task)中开启,并且任务中有且只有这一个实例。若不存在则创建,反之,调用Activity的onNewIntent方法 | 适合和APP分离的且不需要反复创建的界面 |
我们可以反编译一下常用的APP,查看其中的AndroidManifest文件,就能对启动模式有更好的理解了。
<!--网易云音乐的主界面Activity singleTask--!>
<activity android:name="com.netease.cloudmusic.activity.MainActivity" android:launchMode="singleTask" />
<!--百度地图的消息处理Activity singleInstance--!>
<activity android:theme="@style/Theme.NoDisplay" android:name="com.baidu.android.pushservice.opproxy.OpNotifyActivity" android:exported="true" android:launchMode="singleInstance" android:configChanges="keyboardHidden|orientation">
<intent-filter>
<action android:name="com.baidu.BaiduMap.action.RECEIVE_MCS_MESSAGE"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
<!--微信的语音拨号Activity singleTop--!>
<activity android:name="com.tencent.mm.plugin.ipcall.ui.IPCallDialUI" android:launchMode="singleTop" android:screenOrientation="portrait" android:configChanges="keyboardHidden|orientation|screenSize"/>