2020.8.1
一.Application的启动流程
手机开机->应用程序的缩略图标被点击
①点击图标->
②启动ActivityThread->线程的入口点:main函数->创建ActivityThread
③将ActivityThread的对象进行绑定 thread.attach(false, startSeq)
④创建仪表对象,管理程序的生命周期 mInstrumentation = new Instrumentation()->
⑤创建上下文: ContextImpl context = ContextImpl.createAppContext(this,getSystemContext().mPackageInfo);
⑥创建应用程序:
mInitialApplication = context.mPackageInfo.makeApplication(true, null);
app=mActivityThread.mInitialApplication.newApplication(cl,appClass,appContext)
⑦回调application的onCreate方法:instrumentation.callApplicationOnCreate(app)
二.启动Activity的流程:ActivityThread
①handleLaunchActivity->performLaunchActivity
②创建Activity上下文 createBaseContextForActivity
③创建启动的页面
activity = mInstrumentation.newActivity( cl, component.getClassName(), r.intent);
④将界面和窗口Window绑定 activity.attach(appContext,getInstrumentation(),window)
⑤调用onCreate方法布局Activity界面
mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);->
activity.performCreate(icicle, persistentState);->
onCreate(icicle);
所以在Activity的子类中,通过实现onCreate方法,来对界面进行初始化(UI布局,逻辑初始化)
APP启动流程图.png
三.上下文——Context
上下文:运行环境,可以理解为就是一个桥梁,Context提供了调用系统资源的方法,设备Context界面
Activity的context和applicationContext的区别
①大多数情况下,可以相互使用
②如果在涉及界面的跳转,尽量使用Activity的context,因为它提供了默认的任务栈,applicationContext没有提供任务栈
一个Activity的继承于Context,所以通常情况下使用其本身作为Context
四.Activity之间的跳转
Activity就是一个独立的页面
实现页面之间的跳转->Intent意图
显示跳转:同一个应用程序,不同界面之间的跳转,明确指定从哪个页面切换到哪个页面
不传递数据的跳转
val intent=Intent(this,目的界面的java字节码文件)
startActivity(intent) 启动意图,进行界面跳转
//明确指定从当前页面跳转到app页面
val intent=Intent(this,app::class.java)
//启动
startActivity(intent)
finish():结束当前界面,即把当前界面删除,然后从任务栈中调出上一个界面并显示,其实在manifests配置文件中给Activity添加父界面,就是等价于添加了finish()方法
传递数据的跳转
intent.putExtra(Key,Value):通过给intent添加键值对的方式来传递数据
intent.getStringExtra(Key):在目的界面中可以通过intent的键获取对应的值
注:putExtra()->只能传递系统默认的基本类型
intent.putExtra("name","jack")//起始界面传递数据
val name:String?=intent.getStringExtra("name")//目的界面中获取值
val bundle=Bundle()->管理一对键值对
bundle.putString(Key,Value)
intent.putExtras(bundle)同样可以实现传递数据
val bundle=Bundle().apply {
putString("name","jack")
putInt("age",20)
}
intent.putExtras(bundle)
val bundle2=intent.extras:在目的界面中获取Bundle的一个对象
bundle2.getString(Key)获取对应的值
val bundle2:Bundle?=intent.extras
val name=bundle2?.getString("name")
val age=bundle2?.getInt("age")
需要返回值的跳转
startActivityForResult(intent,自定义的请求码)
同时必须重写onActivityResult(requestCode: Int, resultCode: Int, data: Intent?)方法
-----------------------------------起始界面--------------------------------------------
startActivityForResult(intent,1)
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
//1.判断是不是我请求的数据
if (requestCode==1){
//2.判断处理结果 resultCode
if(resultCode==0){
Log.v("cx","处理成功:${data?.getStringExtra("uid")}")
}else{
Log.v("cx","处理失败")
}
}
}
目的界面中setResult(自定义结果码)->不带数据的返回
setResult(自定义结果码,Intent的一个对象)->带数据的返回
返回的数据保存在函数参数data中
-----------------------------------目的界面--------------------------------------------
//1.回调结果 不带数据
setResult(0)
//2.带数据
Intent().apply {
putExtra("uid","001")
setResult(0,this)
}
原界面可以通过data获取返回的数据
隐式跳转:不同程序之间,运行界面的跳转(微信支付、支付宝支付)
打开系统提供的服务(相机、拨号、发消息)
使用方式:由于系统提供的类无法在程序中通过Intent(当前页面,目的页面的字节码文件)来访问,只能通过配置文件的意图过滤器来查找我们需要的Activity,即添加一个action,每个界面有很多个意图过滤器,使用任意一个意图都可以启动该服务,并跳转到服务界面
注:你想要启动系统某个服务界面,如拨号,那么得知道拨号对应的action,你可以通过阅读源代码来查找拨打电话对应的action,也可以通过Intent的静态属性来获取
//拨打电话
Intent().apply {
//通过配置文件的意图过滤器来查找我们需要的Activity
action=Intent.ACTION_DIAL //静态属性获取action
//data传递数据
data=Uri.parse("tel:10086") //传递所需的电话号码,必须按照源码格式
//满足以上两种条件,便可进行隐式跳转
}.also {
startActivity(it)
}
源码解析.png
跳转到其他应用程序
前提:根据启动系统提供的服务可知,隐式跳转到其他系统服务/程序,需要目的Activity提供意图过滤器和传递数据的格式
在应用程序的AndroidManifest.xml中配置
--------------------------------------app1----------------------------------------------
在其他应用中进行跳转
--------------------------------------app2----------------------------------------------
Intent().apply {
action="cx.action.wiwi"
data= Uri.parse("cx:MyAPP")
}.also {
startActivity(it)
}
跳转到app1中,可通过data获取传递的数据
--------------------------------------app1----------------------------------------------
val content=intent.data?.schemeSpecificPart
Log.v("cx",content)