前言
Hi,大家好,上一期我们讲了如何使用BroadcastReceiver,这一期我们讲解Android四大组件之Service相关知识。每天一篇技术干货,每天我们一起进步。
耐心专注不仅仅是美德,更是一笔财富。
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
的方式定义了我们自己的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)
Service和Activity通信
在上面我们高高兴兴的启动了Service
了,但是细心的你可能发现了,貌似我们仅仅只是启动了而已,Activity
跟Service
并没有多少"交流",下面我们就让Activity
跟Service
交流一下。
public class MyService extends Service {
public static final String TAG = "MyService";
private MyBinder mBinder = new MyBinder();
@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 void onDestroy() {
super.onDestroy();
Log.d(TAG, "onDestroy() executed");
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
class MyBinder extends Binder {
public void startDownload() {
Log.d("TAG", "startDownload() executed");
// 执行具体的下载任务
}
}
}
接下来我们在MainActivity
中通过Button
来绑定Service
和解除绑定
public class MainActivity extends Activity implements OnClickListener {
private Button bindService;
private Button unbindService;
private MyService.MyBinder myBinder;
private ServiceConnection connection = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
myBinder = (MyService.MyBinder) service;
myBinder.startDownload();
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
bindService = (Button) findViewById(R.id.bind_service);
unbindService = (Button) findViewById(R.id.unbind_service);
bindService.setOnClickListener(this);
unbindService.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.bind_service:
Intent bindIntent = new Intent(this, MyService.class);
bindService(bindIntent, connection, BIND_AUTO_CREATE);
break;
case R.id.unbind_service:
unbindService(connection);
break;
default:
break;
}
}
}
可以看到,这里我们首先创建了一个ServiceConnection
的匿名类,在里面重写了onServiceConnected()
方法和onServiceDisconnected()
方法,这两个方法分别会在Activity
与Service
建立关联和解除关联的时候调用。在onServiceConnected()
方法中,我们又通过 向下转型 得到了MyBinder
的实例,有了这个实例,Activity
和Service
之间的关系就变得非常紧密了。现在我们可以在Activity
中根据具体的场景来调用MyBinder
中的任何public
方法,即实现了Activity
指挥Service
干什么Service
就去干什么的功能。
当然,现在Activity
和Service
其实还没关联起来了呢,这个功能是在Bind Service按钮的点击事件里完成的。可以看到,这里我们仍然是构建出了一个Intent
对象,然后调用bindService()
方法将Activity
和Service
进行绑定。bindService()
方法接收三个参数,第一个参数就是刚刚构建出的Intent
对象,第二个参数是前面创建出的ServiceConnection
的实例,第三个参数是一个标志位,这里传入BIND_AUTO_CREATE
表示在Activity
和Service
建立关联后自动创建Service
,这会使得MyService
中的onCreate()
方法得到执行,但onStartCommand()
方法不会执行(只有当我们通过 startService()
方法请求启动服务时,调用此方法)。
解除Activity和Service之间的关联,调用
unbindService(connection);
关于销毁Service说明
- 在
MyService
的内部通过stopSelf()
方法来销毁的; - 一个Service必须要在既没有和任何Activity关联又处理停止状态的时候才会被销毁;
- 在Service的onDestroy()方法里去清理掉那些不再使用的资源,防止在Service被销毁后还会有一些不再使用的对象仍占用着内存;
7.IntentService
IntentService
是Service的子类,在介绍IntentService之前,先来了解使用Service时需要注意的两个问题
-
Service
不会专门启动一个线程执行耗时操作,所有的操作都是在主线程中进行的,以至于容易出现ANR,所以需要手动开启一个子线程; -
Service
不会自动停止,需要调用stopSelf()
方法 或者 是stopService()
方法停止;
使用IntentService
不会出现这两个问题,因为IntentService
在开启Service
时,会自动开启一个新的线程来执行它,另外,当Service
运行结束后,会自动停止。
8.如何保证服务不会被杀死
第一种方式,返回 START_STICKY
或 START_REDELIVER_INTENT
当Service
因内存不足而被系统kill后,一段时间后内存再次空闲时,系统将会尝试重新创建此Service
,一旦创建成功后将回调onStartCommand
方法,但其中的Intent
将是null
,除非有挂起的Intent
,如pendingintent
,这个状态下比较适用于不执行命令、但无限期运行并等待作业的媒体播放器或类似的服务。
/**
* 返回 START_STICKY 或 START_REDELIVER_INTENT
* @param intent
* @param flags
* @param startId
* @return
*/
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
//return super.onStartCommand(intent, flags, startId);
return START_STICKY;
}
第二种方式,提高service的优先权
结语
Service作为Android的四大组件之一,并且项目开发过程中一些场景下经常被使用到,小伙伴们赶紧上手实操,把它灵活的运用到项目中,结合上两期的Activity和BroadcastReceiver实现有趣的交互吧。
PS:如果还有未看懂的小伙伴,欢迎加入我们的QQ技术交流群:892271582,里面有各种大神回答小伙伴们遇到的问题哦~