1.简介与定义
简介
Service是一个可以在后台执行长时间运行操作而不提供用户界面的应用组件。Service可由其他应用组件启动,而且即使用户切换到其他应用,Service仍将在后台继续运行。 此外,组件可以绑定到Service,以与之进行交互,甚至是执行进程间通信 (IPC)。 例如,Service可以处理网络事务、播放音乐,执行文件 I/O 或与内容提供程序交互,而所有这一切均可在后台进行。
定义
Service是一个专门在后台处理长时间任务的Android组件。
1.Service不是一个单独的进程;
2.Service也不是一个单独的线程;
3.Service是一个单独的Android组件,Service运行在主线程上,如果想在Service中处理很占时间的操作时,必须在Service中开线程,以降低Activity没有响应的风险;
4.Service不提供用户界面;
它有两种启动方式:startService和bindService。
2.用途
Service有三个常见用途。
1.功能调度:Service接收指定的广播信息,从而进一步分析和处理事件,最后修改数据、更新界面或者进行其他相关的操作,调度整个应用使其保持正确的状态。
2.功能提供:Service并不会接收任何的广播,只接收指定的广播提供状态数据,这时需要绑定Service,绑定Service时要管理好Service,一般在Activity的onStop函数里进行解绑unBindService操作。
3.远程调用:定义AIDL服务,跨进程调用Service,先定义一个远程调用接口,然后为该接口提供一个IBinder实现类,客户端获取了远程的Service的IBinder对象的代理后,通过该IBinder对象去回调远程Service的属性或方法。
3.应用场景
如果某个程序组件需要在运行时向用户呈现界面,或者程序需要与用户交互,就需要用Activity,否则就应该考虑使用Service。
4.Service与Activity对比
相似点:
1.都是单独的Android组件;
2.都拥有独立的生命周期;
3.都是Context的派生类,所以可以调用Context类定义的如getResources()、 getContentResolver()等方法;
4.都拥有自己生命周期回调方法;
不同点:
1.Activity运行于前台有图形用户界面,负责与用户交互;Service通常位于后台运行,不需要与用户交互,也没有图形用户界面。
5.Service的生命周期
随着应用程序启动Service方式不同,Service的生命周期也略有差异,如下图:
如果应用程序通过startService()方法来启动Service,Service的生命周期如上图左半部分所示。
通过调用startService() 方法启动Service: 当其他组件调用startService()方法时,Service被创建,并且无限期运行,其自身必须调用stopSelf()方法或者其他组件调用stopService() 方法来停止Service,当Service停止时,系统将其销毁。
如果应用程序通过bindService()方法来启动Service,Service的生命周期如上图右半部分所示。
通过bindService() 方法启动Service: 当其他组件调用bindService()方法时,Service被创建。接着客户端通过IBinder接口与Service通信。客户端通过unbindService() 方法关闭连接。多个客户端能绑定到同一个Service,并且当他们都解除绑定时,系统将销毁Service(Service不需要被停止)
特别说明:当Activity调用bindService()绑定一个已通过startService()启动的Service时,系统只是把Service内部的IBinder对象传给Activity,并不会把该Service生命周期完全绑定到该Activity,因而当Activity调用unBindService()方法取消与该Service的绑定时,也只是切断该Activity与Service之间的关联,并不能停止该Service组件。要停止该Service组件,还需调用stopService()方法。
在Service的生命周期里,常用的有:
4个手动调用的方法:
手动调用方法 | 作用 |
---|---|
startService() | 启动服务 |
stopService() | 关闭服务 |
bindService() | 绑定服务 |
unbindService() | 解绑服务 |
5个自动调用的方法:
内部自动调用的方法 | 作用 |
---|---|
onCreate() | 创建服务 |
onStartCommand() | 开始服务 |
onDestroy() | 销毁服务 |
onBind() | 绑定服务 |
onUnbind() | 解绑服务 |
6.Service的使用
当我们开始使用Service的时候当然是启动一个Service了,启动Service的方法和启动Activity很类似,都需要借助Intent来实现,下面我们就通过一个具体的例子来看一下。
public class MyService extends Service {
public static final String TAG = "MyService";
@Override
public void onCreate() {
super.onCreate();
Log.d(TAG, "onCreate() executed");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.d(TAG, "onStartCommand() executed");
return super.onStartCommand(intent, flags, startId);
}
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onDestroy() {
super.onDestroy();
Log.d(TAG, "onDestroy() executed");
}
}
要创建一个这样的Service,你需要让该类继承Service类,然后重写以下方法:
onCreate()
1.如果service没被创建过,调用startService()后会执行onCreate()和onStartCommand()方法;
2.如果service已处于运行中,调用startService()不会执行onCreate()方法,只执行onStartCommand()方法。 也就是说,onCreate()只会在第一次创建service时候调用,多次执行startService()不会重复调用onCreate(),此方法适合完成一些初始化工作。
onStartCommand() 如果多次执行了Context的startService()方法,那么Service的onStartCommand()方法也会相应的多次调用。onStartCommand()方法很重要,我们在该方法中根据传入的Intent参数进行实际的操作,比如会在此处创建一个线程用于下载数据或播放音乐等。
onBind() Service中的onBind()方法是抽象方法,Service类本身就是抽象类,所以onBind()方法是必须重写的,即使我们用不到。
onDestroy() 在销毁的时候会执行Service的该方法。
这几个方法都是回调方法,且在主线程中执行,由Android操作系统在合适的时机调用。
注意:每个Service必须在manifest中 通过来声明。
<service android:name="com.demo.service.MyService" >
...
</service>
现在我们通过继承Service的方式定义了我们自己的MyService类,并且在manifest中声明了我们的MyService,接下来我们应该启动我们自己的服务。
启动Service
第一种方式:我们是通过一个Intent对象,并调用startService()方法来启动MyService。
Intent startIntent = new Intent(this, MyService.class);
startService(startIntent);
注意:假如我们是通过点击Button执行上面的代码,那么第一次点击的时候回执行其中的onCreate()跟onStartCommand()方法,但是当我们第二次点击的时候就只会执行onStartCommand()方法。
为什么会这样呢? 这是由于onCreate()方法只会在Service第一次被创建的时候调用,如果当前Service已经被创建过了(第一次点击创建了MyService),不管怎样调用startService()方法,onCreate()方法都不会再执行。
第二种方式:通过bindService启动Service。
bindService启动服务特点: 1.bindService启动的服务和调用者之间是典型的client-server模式。调用者是client,service则是server端。service只有一个,但绑定到service上面的client可以有一个或很多个。这里所提到的client指的是组件,比如某个Activity。
2.client可以通过IBinder接口获取Service实例,从而实现在client端直接调用Service中的方法以实现灵活交互,这在通过startService()方法启动中是无法实现的。
3.bindService启动服务的生命周期与其绑定的client息息相关。当client销毁时,client会自动与Service解除绑定(client会有ServiceConnectionLeaked异常,但程序不会崩溃)。当然,client也可以明确调用Context的unbindService()方法与Service解除绑定。当没有任何client与Service绑定时,Service会自行销毁。
启动了之后,当我们想停止服务的时候该怎么做呢?
停止Service
第一种方式:我们也是通过一个Intent对象,并调用stopService()方法来停止MyService
Intent stopIntent = new Intent(this, MyService.class);
stopService(stopIntent);
第二种方式:调用unbindService(conn)方法来停止MyService
unbindService(ServiceConnection conn)