Android Service总结02 service介绍

版本

版本说明

发布时间

发布人

V1.0

介绍了Service的种类,常用API,生命周期等内容。

2013-03-16

Skywang


概要

若读者之前没接触过service,对下面内容有个大致了解即可。待使用过service之后再来阅读本章内容,会理解更深刻。

service是Android四大组件之一,其余的是activity、BroadcastReceiver、Content Provider。service是一个能够在后台执行长时操作的应用程序组件,它不提供用户界面。应用程序能够通过service来启动后台操作,不同应用程序也能够通过service来实现进程间通信(IPC)。

一个service被启动后,实际上还是运行在主线程中,即UI线程。若用户要执行比较耗时的操作,建议新建一个线程来处理;否则可以到UI线程阻塞,影响用户体验。


1 service种类和区别

这里介绍的android服务有4种,下面分别说明。

1.1 Started Service

即,被启动的服务,它是2种常见服务之一,另一种是Bound Service。它常被用在执行进程的某个后台操作,如通过该服务来实现文件下载等功能。
  要提供被启动的服务:首先,要创建一个继承于Service的类。然后,其它的应用程序可以通过startService()来启动该服务;若要停止服务,可以在应用程序中执行stopService(),也可以在服务中执行stopSelf()来停止服务。

当服务一旦启动,它就能够无限期的在后台运行,即使启动它的组件被销毁。通常,一个被启动的服务执行一个单一操作,并且不给调用者返回结果。若需要返回结果,可以通过广播(BroadcastReceiver)来实现。这些在“Android Service总结03 之被启动的服务 -- Started Service”中以详细示例来说明。

1.2 Bound Service

即,被绑定的服务,和Started Service一样,它也是2种常见服务之一。它也常被用在执行进程的某个后台操作,如通过该服务来实现文件下载等功能。
  要提供被绑定的服务:首先,要创建一个继承于Service的类。然后,其它的应用程序可以通过bindService()来绑定该服务,绑定服务后,就可以调用服务提供的功能;若要解除服务,可以在应用程序中执行unbindService()。

“Started Service”和“Bound Service”的主要区别:

(01) 启动/停止方式不同

  • 若要启动“Started Service”,需要调用startService();要停止该服务,要调用stopService()或stopSelf()。
  • 若要绑定“Bound Service”,则调用bindService();若要解除绑定,则要调用unbindService()。

(02) 实现接口不同

  • “Started Service”中,一般要实现service的onStartCommand()接口。因为startService()会调用onStartCommand()接口,我们一般在onStartCommand()中执行一些服务提供的功能,若启动一个线程用于下载。
  • “Bound Service”中,一般要实现service的onBind()接口。  onBind()函数要返回一个IBinder对象,客户端调用bindService()绑定该服务时被调用。这个绑定是异步的,bindService()方法会立即返回,并且不给客户端返回IBinder对象。要接收IBinder对象,客户端必须创建一个实现ServiceConnection类,并创建ServiceConnection的实例;然后在,bindService()方法中调用ServiceConnection的实例。这里只需要有个大致概念即可;后面在“Android Service总结04 之被绑定的服务 -- Bound Service”中,我们会详细介绍各个接口的作用,以及如何实现这些接口。

(03) 活跃生命周期不同

这里“活跃生命周期”(active lifetime)不同于“完整生命周期”(entire lifetime)。它们的完整生命周期都是起始于onCreat(),结束于onDestroy()。但“活跃生命周期”指的是服务仍然是活动状态。

  • “Started Service”的活跃生命周期开始于startService(),停止于stopService()或stopSelf()。一个“Started Service”可以被多次启动,每次启动都会调用service的onStartCommand()接口;但是,一旦stopService()或stopSelf()被调用,它的活跃生命周期即结束!
  • “Bound Service”的活跃生命周期开始于bindService(),停止于最后一个客户端调用unbindService()。意思是,“Bound Service”可以被多个客户端绑定,即多个客户端可以调用bindService()来绑定相同的服务。当第一个绑定服务的客户端调用bindService()时,“Bound Service”的活跃生命周期即开始了;当所有的客户端全部解除绑定时,它的活跃生命周期即结束了。

  补充:一个service可以同时提供“Started Service”和“Bound Service”。要能提供开始的服务,需实现onStartCommand()接口;要能提供绑定的服务,需实现onBind()。同时实现两种接口,即能提供两种服务。

1.3 IntentService

它的特性是对服务的请求逐个处理,而不能同时处理多个请求。
  要提供IntentService:首先,要创建一个继承于Service的类。然后,其它的应用程序可以通过startService()来启动该服务;若要停止服务,可以在应用程序中执行stopService(),也可以在服务中执行stopSelf()来停止服务,另外,IntentService中的请求被处理完毕,它也会自动停止。

由于“IntentService”和“Started Service”的启动和停止方式一致,我们这里介绍一下它们的主要区别:
(01) 实现原理不同

  • IntentService内部有一个工作队列,逐个对任务进行处理。因为不必担心多线程的问题。
  • “Started Service”支持单线程或多线程,但必须我们自己在服务中实现。

(02) 实现的API不同

  • IntentService必须实现onHandleIntent()接口,在onHandlerIntent()中启动IntentService的Intent进行处理。
  • “Started Service”要实现onStartCommand()接口,服务所需提供的功能,一般在该接口中处理。

(03) 生命周期不同。

  • IntentService中所有的启动请求都被处理之后会自动终止这个服务,因此你不需要自己去调用stopSelf()方法。
  • “Started Service”的生命周期开始于startService(),停止于stopService()或stopSelf()。

1.4 AIDL

AIDL(Android Interface Definition Language),是用来来完成进程间通信(IPC)。
  一般在服务需要接受不同应用多线程的请求时才需要使用AIDL;如果是同一个应用内的请求使用Binder实现即可;如果只是应用间通信而不是多线程处理的话使用Messenger,当然这两种情况也可以使用AIDL。本地进程和远程进程使用AIDL有所不同,本地进程内调用时会都在调用的线程内执行,远程进程使用是通过Service进程内一个由系统维护的线程池发出调用,所以可能是未知线程同时调用,需要注意线程安全问题。


2 服务的API

无论是“Started Service”还是“Bound Service”,它们都是继承与Service类。下面对Service类中的主要API进行简要说明。

onStartCommand():当其它程序调用startService()时,系统会自动调用该函数。且一旦执行了该函数,服务就被启动了。此时,启动的服务是“Started Service”。

onBind():当其它程序调用bindService()时,系统会自动调用该函数。一旦执行了该函数,服务就被启动了。此时,启动的服务是“Bounder Service”。若要提供绑定功能,则必须实现该函数,并且要返回IBinder对象;若不想提供绑定功能,则可以直接返回null。

onUnbind():当其它程序调用unbindService(),系统会自动调用该函数。

onCreate():服务被创建时,系统会自动调用该函数。一般在该函数中进行初始化工作,例如:新建线程。

onDestroy():在服务被销毁时,系统会自动调用该函数。一般在该函数中进行清除工作,例如,终止并回收线程。


3 服务声明方法

在manifest中,通过以下方法声明服务

<manifest ... >
...
<application ... >
<service android:name=".ExampleService" />
...
</application>
</manifest>

像activity一样。你即可以在service对应的manifest中注册intent-filter,让其它程序通过隐式意图来调用该服务;也可以不注册intent-filter,直接通过显式意图来调用该服务。若想让某服务成为某application的私有服务,则可以通过设置android:exported属性为false。


4 服务的生命周期

service 的生命周期时序图如下:

Android Service总结02 service介绍-LMLPHP

更多service内容:

Android Service总结01 目录

Android Service总结02 service介绍

Android Service总结03 之被启动的服务 -- Started Service

Android Service总结04 之被绑定的服务 -- Bound Service

Android Service总结05 之IntentService

Android Service总结06 之AIDL


05-08 15:09