Android之startService与bindService的区别

举报
chenyu 发表于 2021/07/26 23:56:06 2021/07/26
【摘要】 Android执行Service有两种方法,一种是startService,一种是bindService。下面让我们一起来聊一聊这两种执行Service方法的区别。 1、生命周期上的区别     执行startService时,Service会经历onCreate->onStartCommand。当执行stopServic...

Android执行Service有两种方法,一种是startService,一种是bindService。下面让我们一起来聊一聊这两种执行Service方法的区别。

1、生命周期上的区别

    执行startService时,Service会经历onCreate->onStartCommand。当执行stopService时,直接调用onDestroy方法。调用者如果没有stopService,Service会一直在后台运行,下次调用者再起来仍然可以stopService。

    执行bindService时,Service会经历onCreate->onBind。这个时候调用者和Service绑定在一起。调用者调用unbindService方法或者调用者Context不存在了(如Activity被finish了),Service就会调用onUnbind->onDestroy。这里所谓的绑定在一起就是说两者共存亡了。

    多次调用startService,该Service只能被创建一次,即该Service的onCreate方法只会被调用一次。但是每次调用startService,onStartCommand方法都会被调用。Service的onStart方法在API 5时被废弃,替代它的是onStartCommand方法。

    第一次执行bindService时,onCreate和onBind方法会被调用,但是多次执行bindService时,onCreate和onBind方法并不会被多次调用,即并不会多次创建服务和绑定服务。

2、调用者如何获取绑定后的Service的方法

    onBind回调方法将返回给客户端一个IBinder接口实例,IBinder允许客户端回调服务的方法,比如得到Service运行的状态或其他操作。我们需要IBinder对象返回具体的Service对象才能操作,所以说具体的Service对象必须首先实现Binder对象。

3、既使用startService又使用bindService的情况

    如果一个Service又被启动又被绑定,则该Service会一直在后台运行。首先不管如何调用,onCreate始终只会调用一次。对应startService调用多少次,Service的onStart方法便会调用多少次。Service的终止,需要unbindService和stopService同时调用才行。不管startService与bindService的调用顺序,如果先调用unbindService,此时服务不会自动终止,再调用stopService之后,服务才会终止;如果先调用stopService,此时服务也不会终止,而再调用unbindService或者之前调用bindService的Context不存在了(如Activity被finish的时候)之后,服务才会自动停止。

    那么,什么情况下既使用startService,又使用bindService呢?

    如果你只是想要启动一个后台服务长期进行某项任务,那么使用startService便可以了。如果你还想要与正在运行的Service取得联系,那么有两种方法:一种是使用broadcast,另一种是使用bindService。前者的缺点是如果交流较为频繁,容易造成性能上的问题,而后者则没有这些问题。因此,这种情况就需要startService和bindService一起使用了。

    另外,如果你的服务只是公开一个远程接口,供连接上的客户端(Android的Service是C/S架构)远程调用执行方法,这个时候你可以不让服务一开始就运行,而只是bindService,这样在第一次bindService的时候才会创建服务的实例运行它,这会节约很多系统资源,特别是如果你的服务是远程服务,那么效果会越明显(当然在Servcie创建的是偶会花去一定时间,这点需要注意)。    

4、本地服务与远程服务

    本地服务依附在主进程上,在一定程度上节约了资源。本地服务因为是在同一进程,因此不需要IPC,也不需要AIDL。相应bindService会方便很多。缺点是主进程被kill后,服务变会终止。

    远程服务是独立的进程,对应进程名格式为所在包名加上你指定的android:process字符串。由于是独立的进程,因此在Activity所在进程被kill的是偶,该服务依然在运行。缺点是该服务是独立的进程,会占用一定资源,并且使用AIDL进行IPC稍微麻烦一点。

    对于startService来说,不管是本地服务还是远程服务,我们需要做的工作都一样简单。

5、代码实例

    startService启动服务


  
  1. public class LocalService1 extends Service {
  2. /**
  3. * onBind 是 Service 的虚方法,因此我们不得不实现它。
  4. * 返回 null,表示客服端不能建立到此服务的连接。
  5. */
  6. @Override
  7. public IBinder onBind(Intent intent) {
  8. return null;
  9. }
  10. @Override
  11. public void onCreate() {
  12. super.onCreate();
  13. }
  14. @Override
  15. public void onStartCommand(Intent intent, int startId, int flags) {
  16. super.onStartCommand(intent, startId, flags);
  17. }
  18. @Override
  19. public void onDestroy() {
  20. super.onDestroy();
  21. }
  22. }


  
  1. public class LocalService extends Service {
  2. /**
  3. * 在 Local Service 中我们直接继承 Binder 而不是 IBinder,因为 Binder 实现了 IBinder 接口,这样我们可以** 少做很多工作。
  4. */
  5. public class SimpleBinder extends Binder{
  6. /**
  7. * 获取 Service 实例
  8. * @return
  9. */
  10. public LocalService getService(){
  11. return LocalService.this;
  12. }
  13. public int add(int a, int b){
  14. return a + b;
  15. }
  16. }
  17. public SimpleBinder sBinder;
  18. @Override
  19. public void onCreate() {
  20. super.onCreate();
  21. // 创建 SimpleBinder
  22. sBinder = new SimpleBinder();
  23. }
  24. @Override
  25. public IBinder onBind(Intent intent) {
  26. // 返回 SimpleBinder 对象
  27. return sBinder;
  28. }
  29. }

  上面的代码关键之处,在于 onBind(Intent) 这个方法 返回了一个实现了 IBinder 接口的对象,这个对象将用于绑定Service 的 Activity 与 Local Service 通信。

    下面是 Activity 中的代码:


  
  1. public class Main extends Activity {
  2. private final static String TAG = "SERVICE_TEST";
  3. private ServiceConnection sc;
  4. private boolean isBind;
  5. @Override
  6. public void onCreate(Bundle savedInstanceState) {
  7. super.onCreate(savedInstanceState);
  8. setContentView(R.layout.main);
  9. sc = new ServiceConnection() {
  10. @Override
  11. public void onServiceDisconnected(ComponentName name) {
  12. }
  13. @Override
  14. public void onServiceConnected(ComponentName name, IBinder service) {
  15. LocalService.SimpleBinder sBinder = (LocalService.SimpleBinder)service;
  16. Log.v(TAG, "3 + 5 = " + sBinder.add(3, 5));
  17. Log.v(TAG, sBinder.getService().toString());
  18. }
  19. };
  20. findViewById(R.id.btnBind).setOnClickListener(new OnClickListener() {
  21. @Override
  22. public void onClick(View v) {
  23. bindService(new Intent(Main.this, LocalService.class), sc, Context.BIND_AUTO_CREATE);
  24. isBind = true;
  25. }
  26. });
  27. findViewById(R.id.btnUnbind).setOnClickListener(new OnClickListener() {
  28. @Override
  29. public void onClick(View v) {
  30. if(isBind){
  31. unbindService(sc);
  32. isBind = false;
  33. }
  34. }
  35. });
  36. }
  37. }

6、在AndroidManifest.xml里Service元素常见选项  

    android:name  -------------  服务类名

    android:label  --------------  服务的名字,如果此项不设置,那么默认显示的服务名则为类名

    android:icon  --------------  服务的图标

    android:permission  -------  申明此服务的权限,这意味着只有提供了该权限的应用才能控制或连接此服务

    android:process  ----------  表示该服务是否运行在另外一个进程,如果设置了此项,那么将会在包名后面加上这段字符串表示另一进程的名字

    android:enabled  ----------  表示是否能被系统实例化,为true表示可以,为false表示不可以,默认为true

    android:exported  ---------  表示该服务是否能够被其他应用程序所控制或连接,不设置默认此项为 false


文章来源: chenyu.blog.csdn.net,作者:chen.yu,版权归原作者所有,如需转载,请联系作者。

原文链接:chenyu.blog.csdn.net/article/details/51674460

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。