Service及相关知识点总结

说在前面的话

    Service是Android程序里面最常用的基础组件之一,它是一个可以在后台长期运行的组件,用户不能直接
与之交互,但可以通过别的应用组件绑定服务与之交互,service使用需要在manifest中进行配置。
    1. service默认运行在主线程中,处理耗时的逻辑一样回导致ANR。
    2. service创建的目的是Activity不可见或者销毁的情况下仍能执行逻辑。
    3. 一个service无论new多少次只能有一个实例
    4. 可以和一个已经调用startService而被开启的service进行绑定,这种情况下stopService或stopSelf
并不能停止service(除非所有的客户都解除绑定)

1.Service的两种启动方式及生命周期

1).StartedService(生命周期onCreate->onStartCommand->running->onDestroy)

  • 启动
    通过startServce方法启动service,如果不调用stopService或者stopSelf停止服务则可以无限的运行下去(即使启动service的组建被销毁也不影响),第一次启动会调用onCreate和onStartCommand方法,多次启动不再调用onCreate方法只调用onStartCommand方法,并且传入的startId不同。
  • 停止
    通过stopService方法停止service,如果service没启动则没有什么效果;如果service已经通过startService启动,并且没有绑定其他客户则会停止service并且调用onDestroy方法。在service类内可以调用stopSelf方法停止service。

2).BoundService(生命周期onCreate->onBind->running->onUnbind->onDestroy)

  • 启动
    通过bindService方法启动service,会将service与启动的组件进行绑定,生命周期受启动组件什么周期影响,不会无限运行,启动组件可以通过IBinder接口和service进行通信。首次启动会调用onCreate和onBind方法,并且会调用ServiceConnection里的onServiceConnected方法(如果onBind返回null则不会调用),多次绑定不会多次调用onBind方法只得到IBinder对象(同一个),当最后一个绑定调用unBind方法解除绑定才会销毁
  • 停止
    通过unbindService方法解绑service,如果没有bindService(即使startService启动过)会报错崩溃;如果已经bindService则会停止service并调用onUnbind和onDestroy方法

ServiceConnection中onServiceDisconnected方法会在与服务的连接意外中断时(例如当服务崩溃或被终止时)调用该方法。当客户端取消绑定时,系统不会调用该方法。

stopSelf(int startId)与stopSelf()的区别:stopSelf()直接停止;stopSelf(startId)在其参数startId跟最后启动该service时生成的startId相等时才会执行停止服务。startId即是调用onStartCommand时传入的参数。

2.startService和bindService组合使用时的生命周期

    service有两种状态,一种是“已启动”,一种是“已绑定”,当且仅当service没有任何状态时才会销毁
  • start->bind->running->stop->unbind(先start后bind,然后先stop再unbind)

1.start:会调用onCreate和onStartCommand方法;
2.bind:因为service已经启动所以只会调用onBind方法;
3.stop:因为service还别的有客户端bind,所以还有个“已绑定”状态,所以不会被销毁,所以不会执行onDestroy方法;
4.unbind:此时已经没有任何状态了,所以会被销毁,会调用onUnbind和onDestroy方法。

  • start->bind->running->unbind->stop(先start后bind,然后先unbind再stop)

1.start:会调用onCreate和onStartCommand方法;
2.bind:因为service已经启动所以只会调用onBind方法;
3.unbind:因为service是别的客户端通过start启动的,所以还有个“已启动”状态,所以不会被销毁,所以只执行onUnbind方法;
4.stop:此时已经没有任何状态了,所以会被销毁,调用onDestroy方法。

  • bind->start->running->stop->unbind(先bind后start,然后先stop再unbind)

1.bind:会调用onCreate和onBind方法;
2.start:因为service已启动,所以只会调用onStartCommand方法;
3.stop:因为service是别的客户端通过bind启动的,所以还有个“已绑定”的状态,所以不会被销毁,所以不会执行onDestroy方法;
4.unbind:此时已经没有任何状态了,所以会被销毁,会调用onUnbind和onDestroy方法。

  • **bind->start->running->unbind->stop(先bind后start,然后先unbind后stop) **

1.bind:会调用onCreate和onBind方法;
2.start:因为service已启动,所以只会调用onStartCommand方法;
3.unbind:因为service还有别的客户端调用过startService方法,所以还有个“已启动”的状态,所以不会被销毁,所以只会执行onUnbind方法。
4.stop:此时已经没有任何状态了,所以会被销毁,会调用onDestroy方法;

3.Foreground Service(前台服务)

    前台service并不是运行在前台,它也是运行在后台,它只是在service开启时通过使用通知(常驻通知栏,
只有service停止或者通stopForeground来移除)来告诉人们它正在运行,例如音乐播放器通过前台服务来显示
当前播放的曲目、进度条等,这里只介绍简单的使用方法,具体使用方法在后面的RemoteView中再做介绍。
public class MyService extends Service {  

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }  

    @Override
    public void onCreate() {
        super.onCreate();
        PendingIntent contentIntent = PendingIntent.getActivity(this, 0, new Intent(this, MyService.class), 0);
        Notification notification = new Notification.Builder(this)
                .setSmallIcon(R.mipmap.ic_launcher)
                .setWhen(System.currentTimeMillis())
                .setContentTitle("我是标题")
                .setContentText("正在运行")
                .setContentIntent(contentIntent)
                .build();
        startForeground(1, notification);
    }  

    @Override
    public void onDestroy() {
        super.onDestroy();
        stopForeground(true);
    }

}

通过startForeground方法开启,通过stopForeground停止,使用比较简单不再赘述了。

4.IntentService

本部分内容见Android线程—IntentService的使用及原理

5.IPC-Messenger和AIDL的原理及使用(后续更新)

6.总结(后续更新)

你可能感兴趣的:(Service及相关知识点总结)