Service小结

一、简介

Service 是一个可以在后台执行长时间运行操作而不使用用户界面的应用组件。服务可由其他应用组件启动,而且即使用户切换到其他应用,服务仍将在后台继续运行。 此外,组件可以绑定到服务,以与之进行交互,甚至是执行进程间通信 (IPC)。
注意:服务在其托管进程的主线程中运行,它既不创建自己的线程,也不在单独的进程中运行(除非另行指定)。

二、AndroidManifest配置

使用Service必须在AndroidManifest中注册,如下:
<code><manifest ... >
...
<application ... >
<service android:name=".ExampleService" />
</application>
</manifest></code>
可以为service配置一些特定属性。
<code><service android:enabled=["true" | "false"]
android:exported=["true" | "false"]
android:icon="drawable resource"
android:isolatedProcess=["true" | "false"]
android:label="string resource"
android:name="string"
android:permission="string"
android:process="string" >
</service></code>
android:name:属性是唯一必需的属性,用于指定服务的类名。
android:enabled:是否可被系统实例化,默认true.
android:exported : 设为false则其它应用(user ID不同)无法启动此服务。默认true。
android:permission:指定启动服务及其运行所在进程所需的权限。
android:process:指定服务运行进程,默认当前应用进程。
注意:Android5.0以后禁止了隐式声明Intent来启动Service。可以通过setPackage()解决或者把一个隐式Intent转换成显式Intent。

三、Service的生命周期

service_lifecycle.png

<code>public class ExampleService extends Service {
int mStartMode; // 声明服务如果被系统回收之后的行为
IBinder mBinder; // 返回的用于和Service交互的Binder
boolean mAllowRebind; // 声明onRebind是否被使用
@Override
public void onCreate() {
// 服务正被创建
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
// 由于调用了startService(),服务正在启动
// 返回值表明了服务如果被系统回收之后的行为
// 返回值START_NOT_STICKY
// 返回值START_STICKY
// 返回值START_REDELIVER_INTENT
return mStartMode;
}
@Override
public IBinder onBind(Intent intent) {
// 由于调用了bindService(),服务正在绑定
// 返回Binder
return mBinder;
}
@Override
public boolean onUnbind(Intent intent) {
// 服务的所有绑定都调用了unbindService()
// 返回值决定是否允许调用onRebind
return mAllowRebind;
}
@Override
public void onRebind(Intent intent) {
//调用了onUnbind并返回true,再次调用bindService()
}
@Override
public void onDestroy() {
//服务被销毁
}
}</code>
对于长时间运行的服务,系统会随着时间的推移降低服务在后台任务列表中的位置,而服务也将随之变得非常容易被终止。当内存过低且必须回收系统资源以供具有用户焦点的 Activity 使用时,Android 系统就会强制停止服务。
如果将服务绑定到具有用户焦点的 Activity,则它不太可能会终止;
如果将服务声明为前台服务,则它几乎永远不会终止。
如果服务是启动服务,则您必须将其设计为能够妥善处理系统对它的重启。因为如果系统终止服务,那么一旦资源变得再次可用,系统便会重启服务。

四、IntentService

IntentService是Service的子类,创建了默认线程,使用者只需实现onHandleIntent()即可,IntentService具体执行操作如下:
1.创建默认的工作线程,用于在应用的主线程外执行传递给onStartCommand()的所有 Intent。
2.创建工作队列,用于将一个 Intent 逐一传递给onHandleIntent()实现,这样您就永远不必担心多线程问题。
3.在处理完所有启动请求后停止服务,因此您永远不必调用stopSelf()。
4.提供onBind()的默认实现(返回 null)。
5.提供onStartCommand()的默认实现,可将Intent 依次发送到工作队列和onHandleIntent()实现。
代码实现:
<code>public class TestIntentService extends IntentService {
// 必须实现的构造函数
//且需要调用父类的构造函数,传入的参数就是创建的工作线程的名字
public TestIntentService() {
super("word_thread_name");
}
//唯一要实现的方法,方法返回则服务停止
@Override
protected void onHandleIntent(Intent intent) {
}</code>

五、startService

5.1简介

启动服务:startService(Intent intent)
停止服务:stopService(Intent intent);
如果组件通过调用startService()启动服务(这会导致对onStartCommand()
的调用),则服务将一直运行,直到服务使用stopSelf()自行停止运行,或由其他组件Intent会重新传递给Service。

5.2 创建一个Service

<code>public class HelloService extends Service {
private Looper mServiceLooper;
private ServiceHandler mServiceHandler;
class ServiceHandler extends Handler {
public ServiceHandler(Looper looper) {
super(looper);
}
@Override
public void handleMessage(Message msg) {
stopSelf(msg.arg1);
}
}
@Override
public void onCreate() {
HandlerThread thread = new HandlerThread("name");
thread.start();
mServiceLooper = thread.getLooper();
mServiceHandler = new ServiceHandler(mServiceLooper);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Message msg = mServiceHandler.obtainMessage();
msg.arg1 = startId;
mServiceHandler.sendMessage(msg);
return START_STICKY;
}
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onDestroy() {
}
}</code>
每次startService都会回调onStartCommand方法。在这个方法中,我们拥有在运行Service时传递进来的Intent,这样就可以与Service交换一些信息。这个方法需要返回一个整型值。这个整型代表系统应该怎么样处理这个Service:
START_STICKY:使用这个返回值,如果系统杀死我们的Service将会重新创建。但是,发送给Service的Intent不会再投递。这样Service是一直运行的。
START_NOT_STICKY:如果系统杀死了Service,不会重新创建,除非客户端显式地调用了onStart命令。
START_REDELIVER_INTENT:功能与START_STICKY类似。另外,在这种情况下

5.3 前台运行服务

前台服务被认为是用户主动意识到的一种服务,因此在内存不足时,系统也不会考虑将其终止。
通过调用startForeground()设置服务为前台服务,代码如下:
<code>Intent notificationIntent = new Intent(this, ServiceActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);
Notification notification = new NotificationCompat.Builder(this) .setSmallIcon(R.drawable.ic_launcher) .setWhen(System.currentTimeMillis())
.setContentText("content")
.setContentTitle("title")
.setContentIntent(pendingIntent)
.build();
startForeground(1111, notification);</code>
注意:前台服务必须为状态栏提供通知,状态栏位于“正在进行”标题下方,这意味着除非服务停止或从前台删除,否则不能清除通知。

5.4 向用户发送通知

服务可以发送Toast通知和状态栏通知两种。

六、bindService

6.1简介

绑定服务:bindService(Intent intent,ServiceConnection conn,int flags)
解除绑定:unbindService(ServiceConnection conn)
如果组件是通过调用bindService()来创建服务,则服务只会在该组件与其绑定时运行。一旦该服务与所有客户端之间的绑定全部取消,系统便会销毁它。

6.2绑定Service

要想绑定服务,必须:
1.实现 ServiceConnection。
重写两个回调方法:
onServiceConnected()
绑定时调用,传递onBind()的返回值。
onServiceDisconnected()
当服务的连接意外中断时(例如当服务崩溃或被终止时)调用该方法。注意:取消绑定时并不会调用。
2.调用 bindService()
传递ServiceConnection对象
3.调用 unbindService()取消绑定。
注意:当Activity被销毁时,会取消与服务的绑定。但您应该始终在完成与服务的交互时或您的 Activity 暂停时取消绑定,以便服务能够在未被占用时关闭。

6.3扩展Binder

如果服务是自有应用专用,并且在相同的进程中运行,则应通过扩展 Binder 类并从onBind() 返回它的一个实例来创建接口。
以下是具体的设置方法:
1 在您的服务中,创建一个可满足下列任一要求的Binder实例:
包含可调用的公共方法
返回当前Service实例,Service中包含可调用的公共方法
或返回由服务承载的其他类的实例,其中包含可调用的公共方法
2.从onBind()回调方法返回此Binder实例。
3.从onServiceConnected()回调方法中接收Binder

注意:之所以要求服务和客户端必须在同一应用内,是为了便于客户端转换返回的对象和正确调用其 API。服务和客户端还必须在同一进程内,因为此方法不能任何跨进程执行。如果服务只是您的自有应用的后台工作线程,则优先采用这种方法。 不以这种方式创建接口的唯一原因是,您的服务被其他应用或不同的进程占用。

6.4使用Messenger

如需让接口跨不同的进程工作,则可使用 Messenger 为服务创建接口。服务可以这种方式定义对应于不同类型 Message 对象的 Handler。此 Handler 是 Messenger 的基础,后者随后可与客户端分享一个IBinder,从而让客户端能利用 Message 对象向服务发送命令。此外,客户端还可定义自有Messenger,以便服务回传消息。
以下是具体步骤:
1.服务实现一个 Handler,由其接收来自客户端的每个调用的回调
2.Handler 用于创建 Messenger 对象(对 Handler 的引用)
3.Messenger 创建一个 IBinder,服务通过 onBind() 使其返回客户端
4.客户端使用 IBinder 将 Messenger(引用服务的 Handler)实例化,然后使用后者将 Message 对象发送给服务
5.服务在其 Handler 中(具体地讲,是在 handleMessage() 方法中)接收每个 Message
<code>public class MessageService extends Service {
private Messenger message = new Messenger(new MesHandler());
@Override
public IBinder onBind(Intent intent) {
return message.getBinder();
}
class MesHandler extends Handler{
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
}
}
}</code>
这是执行进程间通信 (IPC) 的最简单方法,因为 Messenger 会在单一线程中创建包含所有请求的队列,这样您就不必对服务进行线程安全设计。

6.5使用AIDL

如需直接使用 AIDL,您必须创建一个定义编程接口的 .aidl 文件。Android SDK 工具利用该文件生成一个实现接口并处理 IPC 的抽象类,您随后可在服务内对其进行扩展。
之前采用 Messenger 的方法实际上是以 AIDL 作为其底层结构。Messenger 会在单一线程中创建包含所有客户端请求的队列,以便服务一次接收一个请求。不过,如果您想让服务同时处理多个请求,则可直接使用 AIDL。 在此情况下,您的服务必须具备多线程处理能力,并采用线程安全式设计。

6.6注意事项

1.应该始终捕获 DeadObjectException 异常,它是在连接中断时引发的,也是远程方法引发的唯一异常
2.对象是跨进程计数的引用
3.通常应配对绑定和取消绑定。
例如:在 onStart() 绑定, onStop() 取消绑定。或者在 onCreate() 绑定,onDestroy() 取消绑定。
4.通常情况下,切勿在 Activity 的 onResume() 和 onPause() 期间绑定和取消绑定,因为每一次生命周期转换都会发生这些回调,您应该使发生在这些转换期间的处理保持在最低水平。
5.此外,如果您的应用内的多个 Activity 绑定到同一服务,并且其中两个 Activity 之间发生了转换,则如果当前 Activity 在下一次绑定(恢复期间)之前取消绑定(暂停期间),系统可能会销毁服务并重建服务。 (Activity文档中介绍了这种有关 Activity 如何协调其生命周期的 Activity 转换。)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • 上篇我们讲解了Android中的5中等级的进程,分别是:前台进程、可见进程、服务进程、后台进程、空进程。系统会按照...
    徐爱卿阅读 3,854评论 6 33
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 什么是Service Service是一个可以在后台长时间执行的应用组件。 Service的启动方式 startS...
    MGLEE阅读 359评论 0 0
  • 目录 Service 介绍 Service两种启动方式 使用 测试 IntentService Activity与...
    gaaaaaaaaaao阅读 1,667评论 0 4
  • 辗转又到了阳春三月。 这样的时节,适合去踏青。于是选一个阳光丽日,一个人信步前行,不用去远方,也不用背负行囊,只许...
    请叫我想念熊阅读 392评论 1 5