Android四大组件对Android开发者开说再熟悉不过了,他们是Activity、Service、BroadcastReceiver和ContentProvider。当具有一定开发实践后会发现很多情况情况下,只有对Android体系结构有一定认识,在实际开发中才能写出更加优秀的代码,否者总是只知其表,不知其里,难免有一种雾里看花的感觉。
1.四大组件的运行状态
Android四大组件中BroadcastReceiver既可以在AndroidMenifest.xml中也可以通过java代码完成注册,两者分别称为静态注册和动态注册。而其他三者必须在AndroidMenifest.xml中完成注册。从调用方式来讲,Activity、Service、BroadcastReceiver都需要借助Intent,而ContentProvider不需要借助Intent。
Activity的调用分为显示和隐式两种,显式调用无非就是使用startActivity方法,可以明确地指向一个Activity组件,简单易用不再介绍。隐式调用具体实现在《Android的IntentFilter的匹配规则》中已有详细描述,不在赘述。但隐式Intent指向一个或多个Activity组件,当然也可能没有Activity符合筛选规则。Activity组件的主要作用是展示一个界面并和用户交互,它扮演的其实是一个前台界面的角色。
Service是一种计算型组件,用于在手机后台执行一系列计算任务。Service有两种状态,分为启动状态和绑定状态两种,两者的不同之处在于后者情况下外界可以很方便的和Service组件进行通信,而前者的Service不需要和外界有直接的交互。但需要注意一点,Service本身是处于主线程的,所以耗时的后台计算仍然需要放到单独的线程中完成。
BroadcastReceiver是一种消息型组件,用于在不同的组件乃至不同的应用之间传递消息,他事实上工作在系统内部。静态注册的广播不需要应用启动就可以接收相应的广播。动态广播需要通过Context.registerReceiver()来实现,并且在不需要时使用Context.unRegisterReceiver()来解除广播,这种形式的广播必须启动应用才能完成注册并接收广播。发送和接收过程的匹配通过广播接收者的
ContentProvider是一种数据共享型的组件,用于向其他组件或者其他应用提供数据。ContentProvider提供增删改查四个方法,内部维持一个数据集合,这个集合可以使用数据库实现,也可以使用其他任何类型,没有什么要求。因为这四个方法是在Binder线程池中被调用的,所以处理好线程同步是很重要的。
2.Activity的工作过程
Activity的startActivity()有好几种重载方式,这在我们调用它的时候就会发现,但该方法内部都会调用startActivityForReult方法。下面是截取该方法的部分代码
public void startActivityForResult(Intent intent , int requestCode, @Nullable Bundle options)
{
If(mParent == null){
Instrumentation.ActivityResult ar =
mInstrumentation.execStartActivity(......);
.......
}
.......
}
mParent代表的是ActivityGroup,在API 13之后基本使用Fragment取代ActivityGroup了。下面再看exceStartActivity方法,该方法有一个ApplicationThread类型的参数,而ApplicationThread实际上是ActivityThread的一个内部类,了解Activity启动模式的朋友就会发现ApplicationThread和ActivityThread在Activity的启动过程中多么重。在该方法中有两句重要的代码:
......
int result = ActivityManagerNative.getDefault().startActivity(.......);
checkStartActivityResult(result,intent);
......
第二句很明显是检查启动Activity的结果,如果无法正确的启动一个Activity就会抛出异常错误。启动Activity的时候有一个常见错误“Unable to find explicit activity class;have you declared this activity int your AndroidMenifest.xml”,就是从这抛出的,看到这有疑惑的朋友你是不是豁然开朗了。
既然启动Activity是上面的第一句代码完成的,ActivityManagerNative.getDefault()的具体实现的ActivityManagerService(简称AMS),所以查看一下AMS的startActivity方法即可。在该方法中又绕了一大圈,最终回到AplicationThread的scheduleLaunchActivity()启动Activity。在scheduleLaunchActivity方法中发送一个启动Activity的消息交给Handler处理,这个Handler的名字很简洁,叫H。后面介绍其他三个组件的启动时都会用到H。处理这个启动消息的的时候会先判断是否Application已经创建过了,如果已经创建过了就不会重复创建了,这也就揭示了为什么一个应用只会有一个Application。Application创建完毕后,系统紧接着调用Application的onCreate方法。接着会通过Activity的attach方法来完成一些重要数据的初始化,完成Window的创建并关联Activity,这样当Window接收到外部事件后就可以将事件传递给Activity。最后就是调用Activity的onCreate方法,完成整个启动过程。
3.Service的启动过程
Service有启动和绑定两种状态,值得注意的是,这两种状态是可以共存的,也就是说Service可以同时处于启动和绑定状态。
启动一个Service如下所示:
Intent intentService = new Intent(this,oneService.class);
startService(intentService);
通过绑定的方式启动一个Service:
Intent intentService = new Intent(this,oneService.class);
bindService(intentService,mServiceConnection,BIND_AUTO_CREATE);
Service的启动从ContextWrapper的startService方法开始,该方法又调用了ContextImpl的startService方法。之前已经说过Context的具体实现交给了ContextImpl来完成,ContextImpl的startService源码很复杂,分析Service的启动过程没必要一句一句代码的深究。和Activity的启动类似,最终都是发送消息给Handler H来完成启动。H通过ActivityThread的handleCreateService完成Service的启动。
handleCreateService主要完成下面几件事:
1.通过类加载器创建Service实例;
2.创建Application对象,并调用Application的onCreate方法,当然Application只能创建一次;
3.创建ConTextImpl并通过Service的attach建立两者的关系,注意Service和Actvity都是一个Context;
4.最后调用Service的onCreate方法,并将Service对象保存到ActivityThread的一个ArrayMap列表中。到这里Service已经启动了,下面就是Service的绑定过程。
Service的绑定同样是从ContextWrapper开始的,再最终调用ContextImpl的bandServiceCommon方法。这个方法主要做了下面的两件事:
· 把ServiceConnection对象转化为ServiceDiapatcher.InnerConnection。这是因为服务的绑定可能是跨进程的,所以ServiceConnection必须借助Binder才能让远程服务端回调自己的方法,InnerConnection正好充当Binder这个角色。在ServiceDiapatcher中保存了ServiceConnection和InnerConnection对象。
· bindServiceCommon通过AMS的bindService完成具体的绑定过程。AMS通过一系列的调用,最终调用了ApplicationThread的SchedualBindService,在ApplicationThread中一系列由schedule开头的方法都是通过Handler H来中转的,这次也不例外。在H内部,会交给ActivityThread的handleBindService处理,首先根据Service的token取出Service对象,然后调用Service的onBind方法,onBind方法会返回一个Binder对象给客户端使用。这个时候客户端并不知道已经成功连接Service了,所以需要调用客户端中ServiceConnection的onServiceConnected方法。到这里,Service的绑定过程就算完成了。
4.BroadcastReceiver的工作过程
先回顾一下BroadcastReceiver的使用方法,首先要定义广播的接收者,只需要继承BroadcastReceiver并重写onReceiver方法即可。比如:
public class oneReceiver extends BroadcastReceiver{
@Override
public void onReceiver(Context context,Intent intent){
//onReceiver不能做耗时的工作
Log.e(“......”,intent.getAction()+””);
}
}
定义过接收者后,还需要注册广播接收者:
·静态注册
·动态注册
IntentFilter filter = new IntentFilter();
filter.addAction(“com.ryg.receiver.LAUNCH”);
registerReceiver(new oneReceiver(),filter);
·发送广播
Intent intent = new Intent();
intent.setAction(“com.rgy.receiver.LAUNCH”);
sendBroadcast(intent);
当使用send方法发送广播时,AMS会查找出匹配的广播接收者并把广播发送给他们处理。广播分为普通广播、有序广播和粘性广播。最普遍的还是普通广播。
广播的发送和接受实质是一个过程的两个阶段。在Ansroid5.0中,默认情况不会发送广播给已经停止的应用,实际上在Android3.0就为Intent添加了两个标志位,分别是:
·FLAG_INCLUDE_STOPPED_PACKAGES,表示广播会发送给已经停止的应用
·FLAG-EXCLUDE_STOPPED_PACKAGES,表示广播不会发送给已经停止的应用
系统默认为Intent添加了第二种标志位,以免广播无意间或者唤起不必要的应用。如果确实要唤起已经停止的应用,为广播添加上面的第一种标志位即可。当两种标志位共存时,广播会发送给已经停止的应用。而最终广播发送的实现还是在前面提到过的Handler H类中完成。
5.ContentProvider的工作过程
ContentProvider是一种内容共享型组件,他通过Binder向其他组件乃至其他应用提供数据。当ContentProvider所在的进程启动时,ContentProvider会同时启动,并且要注意ContentProvider的onCreate是优先于Application的onCreate执行的。这和前面介绍的三个组件是完全不同的。
当一个应用启动时,入口方法为ActivityThread的main方法,在main方法中会创建ActivityThread的实例并创建主线程的消息队列。在ActivityThread的handleBandApplication方法中,ActivityThread会创建Application对象并加载ContentProvider。这里要注意的是,ActivityThread会先加载ContentProvider再调用Application的onCreate方法。流程图如下:
ContentProvider是否是单实例是由他的multiprocess属性决定的。当android:multiprocess为false,ContentProvider是单实例的,当他是true的时候,ContentProvider就是多实例。多实例是指每个调用者的进程都会有一个ContentProvider对象,但这样会带来进程间通信的开销,所以大部分还是使用单实例。
当ContentProvider所在进程还未启动时,第一次访问他会出发ContentProvider的创建,当然也伴随着ContentProvider所在进程的启动。ContentProvider四个方法中的任何一个都会触发启动ContentProvider,下面以查询query为例。
在query方法中会调用ActivityThread的acquireProvider方法,该方法会先从ActivityThread中查找是否已经存在目标ContentProvider了,如果存在直接返回。如果目前ContentProvider还没启动,那么久发送一条进程间请求给AMS让其启动ContentProvider。启动ContentProvider的时候会先启动它所在的进程(也可以理解为是应用),然后再启动当前进程的ContentProvider。AMS会把ContentProvider存储在ProviderMap,这样一来调用者就可以直接从AMS中获取ContentProvider了。