Android应用框架之Service

之前的博客已经介绍了应用框架中的ActivityApplication,今天来讲四大组件之一的Service。对于Service大家肯定都比较熟悉,与Activity最大的不同就是Service不会与界面打交道,而是始终工作在后台,执行一些与UI无关的操作和计算。即便用户切换了其他应用,启动的Service仍可在后台运行。一个组件可以与Service绑定并与之交互,甚至是跨进程通信(IPC)。
Service运行在主线程中(A service runs in the main thread of its hosting process),Service并不是一个新的线程,也不是新的进程。也就是说,若您需要在Service中执行较为耗时的操作(如播放音乐、执行网络请求等),需要在Service中创建一个新的线程。这可以防止ANR的发生,同时主线程可以执行正常的UI操作。
Service有两种启动方式,一个是startService,一个是bindService,接下来分别介绍一下两种方式的启动逻辑。

1.startService

通常情况下启动一个Service的代码如下:

Intent intent = new Intent(this, MyService.class);
context.startService(intent);

启动过程是从Context开始的,而这个Context实际是一个ContextWrapper,而从ContextWrapper的实现看来,其内部实现都是通过ContextImpl来完成的,这是一种典型的桥接模式。通过调用ContextImplstartService,会启动一个服务,核心代码如下所示:

private Component startServiceCommon(Intent service, UserHandle user) {
......
Component cn = ActivityManagerNative.getDefault().startService(mMainThread.getApplicationThread(),service,service.resolveTypeIfNeeded(getContentResolver()),user.getIdentifier());
......

ContextImpl通过ActivityManagerNative.getDefault()获取到一个服务,这个服务就是熟悉的Activity Manager Service(AMS),启动这个服务的方式当然还是Binder机制。所起启动Service的工作就转移到了AMS身上。在AMS的内部还有一个mServices,这个对象是辅助AMS进行service管理的类,包括Service的启动、绑定和停止等等。同时一个Service在AMS内部对应一个ServiceRecord,AMS用它来记录各个Service。
而在AMS内部会通过realStartServiceLocked方法来启动Service,其实在AMS内部的启动步骤还有还经过了很多方法,不过最为核心的就是realStartServiceLocked,该方法的核心代码如下:

private final void realStartServiceLocked(ServiceRecord r, ProcessRecord app, boolean execInFg) throws RemoteException {
......                  app.forceProcessStateUpTo(ActivityManager.PROCESS_STATE_SERVICE);
app.thread.scheduleCreateService(r, r.serviceInfo, mMm.compatibilityInfoForPackageLocked(r.serviceInfo.applicationInfo), app.repProcState);
r.postNotification();
......
}

这里的app是一个ProcessRecord对象,就是在之前的博客中提到的AMS中用于记录一个Application的对象。通过app.thread.scheduleCreateService方法来创建Service并调用其onCreate方法,接着在通过sendServiceArgsLocked方法来调用Service的其他方法,比如onStartCommand。而这两个过程均是进程间通信,app.thread其实是一个IApplicationThread类型,实际就是一个Binder。而scheduleCreateService就是这个binder中的一个接口方法,接下来看一下对应的scheduleCreateService方法:

public final void scheduleCreateService(Binder token, ServiceInfo info, CompatibilityInfo compatInfo, int processState) {
updateProcessState(procesState, false);
CreateServiceData s = new CreateServiceData();
s.token = token;
s.info = info;
s.compatInfo = compatInfo;

sendMessage(H.CREATE_SERVICE, s);
}

从代码中可以看到,最后的创建工作又通过发送消息给Handler H将创建Service的工作又回到了ActivityThread中。最后再来看看ActivityThread的handleCreateService

private void handleCreateService(CreateServiceData data){
......
Service service = null;
java.lang.ClassLoader cl = packageInfo.getClassLoader();
service = (Service)cl.loadClass(data.info.name).newInstance();
......
ContextImpl context = ContextImpl.createAppContext(this, packageInfo);
context.setOuterContext(service);

Application app = packageInfo.makeApplication(false, mInstrumentation);
service.attach(context, this, data.info.name, data.token, app, ActivityManagerNative.getDefault());
service.onCreate();
mServices.put(data.token, service);
......
}

这个方法主要做了以下几件事:

1.通过创建类加载器,创建Service实例
2.创建Application对象,并调用其onCreate方法,当然Application对象只会被创建一次
3.创建ContextImpl对象,并通过service的onAttach方法建立两者之间的联系。这个过程和Activity类似,毕竟Activity和Service都是一个Context
4.最后调用Service的onCreate方法,并将Service保存在ActivityThread中的一个列表mServices。

由于Service的onCreate方法被执行了,接下来AcitivtyThread还会通过handleServiceArgs方法调用Service的onStartCommand方法:

private void handleServiceArgs(ServiceData data) {
Service s = mServices.get(data.token);
......
if(!data.taskRemoved) {
    res = s.onStartCommand(data.args, data.flags, data.startId);
} else {
    s.onTaskRemoved(data.args);
    res = Service.START_TASK_REMOVED_COMPLETE;
}
......
ActivityManagerNative.getDefault().serviceDoneExecuting(data.token, 1, data.startId, res);
......
}

在这个方法中可以看到,在service执行完成之后,还会通过ActivityManagerNative.getDefault().serviceDoneExecuting来通知AMS service已经执行完毕。
最后来总结一下Service启动的主要步骤:

Context–>AMS–>app.thread–>ActivityThread–>Service

为什么要去绕这么一大圈呢?其实很好理解,AMS管理各个组件,要创建一个新的service当然要通过AMS来维护一个与该service对应的实例并与对应的进程实现关联,app.thread只是一个应用通信的接口,并将对应的工作交接给ActivityThread,ActivityThread才是应用的真正实例,它当然也要管理该Service,并维护一个对应的记录(mServices)。其实Activity和Service的启动过程大致相同,从中可以更加了解Android的应用框架。

2.bindService

bindService的大致过程过程和startService类似,还是通过contextImpl.bindServiceCommon来启动:

private boolean bindServiceCommon(Intent service, ServiceConnection conn, int flags, UserHandle user) {
IServiceConnction sd;
......
sd = mPackageInfo.getServiceDispatcher(conn, getOuterContext(), mMainThread.getHandler(), flags);
......
int res = ActivityManagerNative.getDefault().bindService(mMainThread.getApplicationThread(), getActivityToken(), service, service.resolveTypeIfNeeded(getContentResolver()), sd, flags, user.getIdentifier());
......
}

这里主要做了两件事:

  • 将客户端的ServiceConnection转化为ServiceDispatcher.InnerConnection。之所以不能直接使用ServiceConnection是因为绑定的服务可能是跨进程的,所以必须借助于Binder才能让远程服务回调自己的方法。而ServiceDispatcher的内部类InnerConnction正好充当了这个Binder。所以ServiceDispatcher的作用就是ServiceConnection和InnerConnection连接的桥梁。
  • 调用AMS的bindService方法来完成Service的具体绑定过程。

接下来重点讲一下AMS的bindService方法。和startService方法类似的是,bindService最终会将调用到app.thread.scheduleBindService():

public final void scheduleBindService(Binder token, Intent intent, boolean rebind, int processState) {
    updateProcessState(processState, false);
    BindServiceData s = new BindServiceData();
    s.token = token;
    s.intent = intent;
    s.rebind = rebind;
    ......
    sendMessage(H.BIND_SERVICE, s);
}

接下来又转移到了ActivityThread中,而这个方法就是ActivityThread.handleBindService()

private void handleBindService(BindServiceData data) {
    Service s = mServices.get(data.token);
    ......
    IBiner binder = s.onBind(data.intent);               ActivityManagerNative.getDefault().publishService(data.token, data.intent, binder);
    ......
}

handleBindService中,首先根据Service的token取出Service对象,然后调用Service的onBind方法。但是onBind方法是Service的方法,这个时候客户端并不知道已经绑定成功了,所以还必须调用客户端的ServiceConnection中的onServiceConnected,这个是由ActivityManagerNative.getDefault().publishService方法来完成的。最终指令流会转移到mServices(AMS内部的辅助Service)的publishServiceLocked。其核心代码只有一行:c.conn.connected(r.name, service),其中c.conn类型是ServiceDispatcher.InnerConnection,service就是Service的onBind返回的Binder对象。接下来看看ServiceDispatcher.InnerConnection的定义:

private static class InnerConnection extends IServiceConnection.Stub {
    ...
    private void connected(ComponentName name, Binder service) throws RemoteException {
        LoadedApk.ServiceDispatcher sd = mDispacher.get();
        if(sd != null) {
            sd.connectd(name, service);
        }
    }
}

InnerConnection最后通过ServiceDispatcher的connected方法来调用ServiceConnection的onServiceConnected,至此绑定完成。

时间: 2024-11-05 12:12:28

Android应用框架之Service的相关文章

全方位解读Android多媒体框架源码

Android中对于图形界面以及多媒体的相关操作比较容易实现.而且对于大多数手机用户来说,他们主要也就是根据这些方面的功能来对系统那个进行修改.我们可以通过本文介绍的Android多媒体框架的源码解读,来具体分析一下这方面的基本知识. Android多媒体框架的代码在以下目录中:external/opencore/.这个目录是Android多媒体框架的根目录,其中包含的子目录如下所示: * android:这里面是一个上层的库,它基于PVPlayer和PVAuthor的SDK实现了一个为Andr

《Android和PHP开发最佳实践》一2.3 Android应用框架

2.3 Android应用框架 前面介绍了Android的系统框架,主要目的是让大家对Android系统有整体的概念,也为日后更深入的学习打好基础.然而,目前我们更需要重点学习和掌握的则是Android的应用框架,因为是否能掌握和理解Android应用框架,直接关系到是否能学好Android应用开发. Android的应用框架是一个庞大的体系,想要理解透彻并不是那么简单的事情,但是,好在其中有一些比较清晰的脉络可以帮助我们快速地熟悉这个系统,因此抓住这些脉络中的核心要点对于能否学好Android

Android应用框架之应用启动过程详解_Android

在Android的应用框架中,ActivityManagerService是非常重要的一个组件,尽管名字叫做ActivityManagerService,但通过之前的博客介绍,我们知道,四大组件的创建都是有AMS来完成的,其实不仅是应用程序中的组件,连Android应用程序本身也是AMS负责启动的.AMS本身运行在一个独立的进程中,当系统决定要在一个新的进程中启动一个Activity或者Service时就会先启动这个进程.而AMS启动进程的过程是从startProcessLocked启动的. 1

《Android和PHP开发最佳实践》一2.2 Android系统框架

2.2 Android系统框架 在开始介绍Android应用开发之前,我们先来了解一下Android的系统框架.虽然,是否了解Android系统框架与能否进行Android应用开发之间没有任何必然的联系,但是在学习Android的过程中,这个部分内容却是必不可少的,因为能否理解Android的系统架构对于你日后能否对Android进行更深入的学习是至关重要的.首先,我们来看一张不得不说的图,也就是Google官方公布的Android的系统框架图,如图2-1所示.从图2-1展示的Android系统

Android Xutils框架使用问题及解决办法

    刚刚写了篇博客,提了下在使用XUtils时遇到的一个问题Android Xutils框架HttpUtil Get请求缓存问题 ,既然已经提起来这个问题,那我想了下,就把之前使用Xutils时遇到的几个小问题整理一下. 一. HttpUtil Get请求缓存问题       关于Xtuls框架的Http模块的get请求方法,会有缓存问题--即进行get请求时,在缓存规定的时间内,如果发送同样的请求url,则不会再次请服务器发送请求,而是直接返回上次请求的结果. 之前写了一篇博客已经介绍的很

关于android vitamio框架的问题,用过的大神来

问题描述 关于android vitamio框架的问题,用过的大神来 在小屏状态下怎么让MediaController(那个进度条)在视频中下方(下端对其) ![![图片说明](http://img.ask.csdn.net/upload/201509/14/1442231117_275306.png)图片说明](http://img.ask.csdn.net/upload/201509/14/1442231108_654382.png) 解决方案 特意去看了下,不知道你用的什么版本,Media

Android Volley框架全面解析_Android

 Volley简介 我们平时在开发Android应用的时候不可避免地都需要用到网络技术,而多数情况下应用程序都会使用HTTP协议来发送和接收网络数据.Android系统中主要提供了两种方式来进行HTTP通信,HttpURLConnection和HttpClient,几乎在任何项目的代码中我们都能看到这两个类的身影,使用率非常高. 不过HttpURLConnection和HttpClient的用法还是稍微有些复杂的,如果不进行适当封装的话,很容易就会写出不少重复代码.于是乎,一些Android网络

Android Volley框架使用源码分享_Android

过去在Android上网络通信都是使用的Xutils 因为用它可以顺道处理了图片和网络这两个方面,后来发觉Xutils里面使用的是HttpClient  而Google在6.0的版本上已经把HttpClient废除了,所以开始寻找新的网络框架,okhttp也用过,但是它是在作用在UI线程,使用起来还需要用handler 所以就先用着Volley框架了.  这里我先分析下Volley框架的简单网络请求的源码. 使用Volley请求网络数据的简单过程: RequestQueue queue = Vo

数据库-使用Android ORMlite框架时候出现异常

问题描述 使用Android ORMlite框架时候出现异常 主要界面如下 想要向数据库中存入一条数据 BEAN类 package com.newtouch.bean; import com.j256.ormlite.field.DatabaseField; import com.j256.ormlite.table.DatabaseTable; @DatabaseTable(tableName="tb_info") public class User { @DatabaseField