天天看点

android启动器开发详解,Android开发——详解Activity

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布局,逻辑初始化)

android启动器开发详解,Android开发——详解Activity

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)

}

android启动器开发详解,Android开发——详解Activity

源码解析.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)