我们都知道 Service 是 Android 四大组件之一,而 Service 是什么呢?一种可以在后台执行长时间运行操作而没有用户界面的组件。
今天就从这几个方面来说说 Service:
- 与 Thread 做对比
- Service 的两种启动方式
- Service 的生命周期
- Service 的跨进程使用
- 其他
与 Thread 做对比
如果看了我写的 Service 是什么会不会让人觉得和线程有些类似呢?但其实它们本质上就不一样,Service 其实是运行在主线程的,而线程本身的运行就是另起了一个线程,它们所在的范围也不一样,Service 作为远端全局工作,可以和任何 Activity 进行交互,而 Thread 则只在被创建的地方被使用,如果看生命周期的话,Service 有自己的声明周期方便管理,而 Thread 则没有明显的生命周期,只是有着四种状态(新建、就绪、阻塞、死亡),且他们与进程的关系也不同,Service 本身可以指定运行的进程,且可以通过 Binder 进行跨进程通信,如果在单独的进程中运行,还会影响进程的优先级,而 Thread 则是进程的一部分,它只属于某个进程,并不会影响进程优先级。所以其实它们根本就是两个不一样的东西。
Service 的两种启动方式
说了这么多 Service 与 Thread 的内容,下面我们来 Service 的两种方式:
- startService() / stopService()
- bindService() / unbindService()
两个其实还是有很明显的区别的,前者只是启动了 Service,而后者在调用时就需要传递一个 ServiceConnection,而它会返回给我们 IBinder(Binder / Message),便于我们通过它进行 Activity 与 Service 进行交互。
Service 的生命周期
说到 Service 的生命周期,其实与它的启动方式有关,不同的启动方式,调用的声明周期也不同,下图就是官网给出的 Service 的生命周期图,左图显示了使用 startService() 所创建的服务的生命周期,右图显示了使用 bindService() 所创建的服务的生命周期。
不过这里有几点需要注意的是:
1.多次使用 startService() 启动服务,只有第一次会调用 onCreate() 与 onStartCommand(),此外都只会调用 onStartCommand()。
2.通过 bindService() 启动时,Service 可与多个 Activity 进行绑定,而如果要销毁这个服务,就需要解绑所有 Activity。
3.同时调用两种启动方式,那么只有再通过两种方式的停止服务后,服务才会销毁。
Service 的跨进程使用
为什么要使用多进程呢?
- 进程隔离
- 内存均衡
- 重要进程降低内存占用(提高进程优先级,避免进程被系统杀掉)
所以处于某种原因当我们需要自己的 Service 在新的进程中运行时,我们需要通过在清单文件声明 Service 时使用 android:process="..."。
不过很多时候还需要在 Service 跨进程运行的时候与某个 Activity 进行交互,这个时候涉及到跨进程通信,跨进程通信其实其实就需要 Binder 来实现,而 Binder 对象需要我们自己创建,并调用 bindService() 启动后通过 ServiceConnection 返回 Binder 对象进行通信。
基本上我们都是使用自定义的 AIDL,然后通过 AIDL 生成对应的代码,然后进行使用,这里不具体说如何具体操作,不过我前两个刚刚写了一个音乐播放器的小 Demo,里面涉及到 Service 的跨进程通信,代码地址如下:
其他
1.使用 startService() 启动 Service 会调用这个生命周期方法 onStartCommand(),而这个方法返回值需要稍稍注意:
- START_NOT_STICKY:如果 Service 被系统杀掉,则该 Service 不会重启。
- START_STICKY:如果 Service 被系统杀掉,则会在稍后被重启,但不会重传 Intent。
- START_REDELIVER_INTENT:如果 Service 被系统杀掉,则会在稍后被重启,且会重传 Intent。
2.正常来说 Service 运行在主线程且对于停止服务只能主动调用对应的方法,而系统提供了一个封装好的 Service——IntentService,它可以把任务放在工作中线程去完成,而且会在任务完成后主动停止服务,这里不细讲它,而且它的源码很简单,看过了你就明白了。
3.如何提高进程的优先级:
- 减少内存使用,及时释放内存。
- 将重要的进程与普通 UI 进程分离。
- 使用 Service 并使用 startForeground() 提高优先级。
其实这篇只是大概的介绍了一些 Service 的知识点,如果了解更多关于 Service 的知识点,可以去看官方的文档,Service 类与 Service 的 API 指导。