Service
服务是Android中实现程序后台运行的解决方案。
主要用于在后台处理一些耗时的逻辑,或者某些需要长期运行的任务,比如下载。
服务依赖于创建服务时所在的应用程序进程,当应用程序进程被关掉后,所有依赖于该进程的服务也会停止运行。
服务中的代码也是运行在主线程中的。
1.创建一个服务
通过Android Studio来创建服务
右键包名然后 New>Service>Service
创建时有两个属性可以勾选,
Exported属性表示其他程序访问这个服务,Enabled属性用于表示是否启用这个服务。
注意:每个服务都需要在AndroidManifest.xml中进行注册才能生效。Android Studio 已经帮我们创建好了。
这个是创建好的类: 关于onBind()方法,之后再活动和服务之间的通信会用到
public class MyService extends Service { public MyService() { } @Override public IBinder onBind(Intent intent) { // TODO: Return the communication channel to the service. throw new UnsupportedOperationException("Not yet implemented"); }
Service 中常用的几个方法:
- onCreate() :服务创建时调用
- onStartCommand() :服务每次启动时调用
- onDestroy() :服务销毁时调用 ,一般用于回收资源
2.开启和关闭服务
- 开启服务:
Intent startIntent=new Intent(MainActivity.this,MyService.class); startService(startIntent);
首次启启动时会创建一个Service实例,并一次调用onCreate()和onStartCommand()方法进入运行状态,如果再次调用StartService()启动Service,将不会再创建新的Service对象,而是会重用前面创建的Service对象,并调用onStartCommand()方法。
- 关闭服务:
Intent stoptIntent=new Intent(MainActivity.this,MyService.class); stopService(stoptIntent);
另外也可以通过在服务中调用 stopSelf()
来关闭服务
3.Service和Activity之间的通信
我们需要通过Service中的 onBind()方法来实现通信。
比如说我们要实现下载功能,我们需要创建一个内部Binder类,并通过内部类对象来对下载进行管理
public class MyService extends Service { private DownLoadBinder mBinder=new DownLoadBinder(); @Override public IBinder onBind(Intent intent) { return mBinder; } class DownLoadBinder extends Binder{ public int progress=0; public void startDownLoad(){ new Thread(new Runnable() { //在子线程中执行 @Override public void run() { Log.d("MyService", "startDownload executed"); while (progress <= 100) { try { //模拟耗时操作 Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } progress++; } } }).start(); } public int getProgress(){ return progress; } }}
然后在MainActivity中需要修改的是:
创建ServiceConnection匿名类和自定义的Binder类,并通过向下转型得到DownLoadBinder的实例
private MyService.DownLoadBinder downLoadBinder; //监听活动和服务的连接状况,如果成功则回调onServiceConnected() private ServiceConnection connection=new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) { Log.d(TAG, "onServiceConnected: "); downLoadBinder=(MyService.DownLoadBinder)service; downLoadBinder.startDownLoad();//开始下载 } @Override public void onServiceDisconnected(ComponentName name) { Log.d(TAG, "onServiceDisconnected: "); } };
- 绑定服务:
Intent bindIntent=new Intent(MainActivity.this,MyService.class); bindService(bindIntent,connection,BIND_AUTO_CREATE);
但三个参数BIND_AUTO_CREATE,表示在活动和服务进行绑定之后自动创建服务,
这样会使服务中的onCreate()执行,但是startCommand()不会执行,
- 解绑服务:
unbindService(connection);
当解除绑定后服务也就紧跟着销毁了
接下来我们就可以调用 downLoadBinder 的相关方法进行操作,实现在活动中调用服务中方法
比如 获取当前下载进度:
downLoadBinder.getProgress()
注:服务可以跟整个应用程序中任意一个Activity进行绑定,并获取Binder实例。
4.服务的生命周期
注:图片来自 菜鸟教程 http://www.runoob.com/w3cnote/android-tutorial-service-1.html当我们对一个服务即调用了startService()方法,又调用了bindService()方法
Android 系统的机制是,一个服务只要被启动或者被绑定后,就会一直处于运行状态,必须要两种状态同时不满足,服务才会销毁。
所以我们需要通过同时调用stopService()和unbindService()方法来销毁这个服务。
5.前台服务
服务的系统优先级相对来说还是比较低的,所有当系统出现内存不足的情况下,就有可能会回收掉正在后台运行的服务。当我们需要服务一直保持运行状态时,就可以考虑使用前台服务。
前台服务和普通服务的最大区别就是,前台服务会在状态栏一直显示一个通知。常见的有(网易云音乐、360...)
@Override public void onCreate() { //前台服务 Intent intent=new Intent(this,MainActivity.class); PendingIntent pi=PendingIntent.getActivities(this,0,new Intent[]{ intent},0); notificationBuilder= new NotificationCompat.Builder(this); notificationBuilder.setContentTitle("Download") ; //标题 notificationBuilder.setWhen(System.currentTimeMillis()); //通知被创建的时间 notificationBuilder.setSmallIcon(R.mipmap.ic_launcher); //状态栏通知图标 notificationBuilder.setProgress(100,1,false); //条形进度条显示 notificationBuilder.setAutoCancel(true); notificationBuilder.setContentIntent(pi); //点击通知事件 //发送通知,并将服务变为前台服务 startForeground(1,notificationBuilder.build()); }
接着我们可以调用setProgress()方法来更新进度条
notificationBuilder.setProgress(100,progress,false); notificationBuilder.setContentText(progress+"/100");
重点就是startForeground() ,这个方法会将服务变为前台服务,并在系统状态栏中显示出来。
6.IntentService
IntentService可以简单地创建一个异步的、会自动停止的服务。
创建MyIntentService
public class MyIntentService extends IntentService { private static final String TAG = "MyIntentService"; public MyIntentService(){ super(""); //必须调用父类的有参构造函数,否则会抛异常 } @Override protected void onHandleIntent(Intent intent) { Log.d(TAG, "onHandleIntent: "); Log.d(TAG,"Thread name :"+ Thread.currentThread().getName()); } @Override public void onDestroy() { Log.d(TAG, "onDestroy: "); super.onDestroy(); }}
最后不要忘了在AndroidManifest.xml中进行注册服务
启动服务:
Intent intentService=new Intent(this,MyIntentService.class); startService(intentService);
注意:
IntentService 运行完毕后是自动停止。
MyIntentService 中必须调用父类的有参构造函数,否则会抛异常