Android Service调用流程解析

作者: 酱爆大头菜 | 来源:发表于2019-11-06 18:58 被阅读0次

Service是面试中经常问到的一个模块,从之前的简单使用,到后期的应用保活,当然现在Android系统对这种保活机制进行了优化和限制,我们先简单了解下Service的特性。

接下来的文章主要以一问一答的方式进行介绍Service的特性,必要部分也会分析到Service的源码逻辑

首先啥是Service?
  • 虽然大家都知道啥是Service,但是我也得甩个定义。
  • Service是Android系统中的四大组件之一,主要有两个应用场景:后台运行和跨进程访问。Service可以在后台执行长时间运行操作而不提供用户界面,除非系统必须回收内存资源,否则系统不会停止或销毁服务。服务可由其他应用组件启动,而且即使用户切换到其他应用,服务仍将在后台继续运行。 此外,组件可以绑定到服务,以与之进行交互,甚至是执行进程间通信 (IPC)。
Service启动方式有几种?
  • 启动方式有两种。
    1. startService(Intent)
    1. bindService(Intent,ServiceConnection,flags);
startService和bindService启动服务有什么区别?
    1. 这两种启动方式的生命周期不同。
      startService:onCreate()-onStartCommand()-onDestroy();
      bindService:onCreate()-onBind()-onServiceConnected()-onUnBind()-onDestroy();
      混合调用,先startService后bindService
      onCreate()-onStartCommand()-onBind()-onServiceConnected()-onUnBind()-onDestroy()
      混合调用,先bindService后startService
      onCreate()-onBind()-onServiceConnected()-onStartCommand()-onUnBind()-onDestroy()
    1. 启动者结束后,使用startService启动的Service会继续执行,因为startService是独立进程,使用bindService启动的服务会立即结束并抛异常提醒。
    1. startService独立进程 资源占用也会比较多。
多次调用startService或者bindService会重复执行生命周期么?
    1. 重复startService会重复执行onStartCommand(),但onCreate()只会执行一次。
    1. 重复bindService所有Service的生命周期都不重复执行,onCreate(),onBind()只执行一次。
  • 补充:通过tartService和bindService混合启动的Service至少点击一次stopService和unbindService。

Service 怎么和Activity进行通信?
    1. 可通过Binder对象进行通信。
    1. 可通过BroadCast Receiver
    1. 通过Handler借助Binder
    1. 通过回调借助Binder
为什么bindService可以跟Activity生命周期联动?
  • 这个问题字面是在考察Service的联动现象,实在考察对Android源码的理解,接下来咱们就针对源码捋一遍。
    首先我们知道,bindService 方法执行时,会回调到onServiceConnected()。

  • 那onServiceConnected是怎么回调的?又是由谁回调的?
    Service的大部分操作均由ActivityThread的内部类H处理。

  • onServiceConnected()调用链如下所示:

H->handleMessage()
    handleBindService()
          IBinder binder = mServices.get(data.token).onBind();
          // ActivityManager.getService()得到的是一个ActivityManagerService对象。
          ActivityManager.getService().publishService(,, binder);
           //ActivityManagerService对象是在系统进程中,所以当前线程会挂起,启用系统进程调用publishServiceLocked()
              ActiveServices.publishServiceLocked(,, binder);
                //c.conn.connected又从系统的服务进程回到了应用进程,connected方法就是LoadApk的内部类InnerConnection.connected()方法.
                  c.conn.connected(r.name, binder, false); 
                      LoadedApk.ServiceDispatcher.connected(, binder, );
                          LoadedApk.ServiceDispatcher.doConnected(, binder, );
                              ServiceConnection.onServiceDisconnected();
                              ServiceConnection.onServiceConnected();
  • 以上执行过程中LoadedApk会记录 ServiceConnection 信息。
    当Activity 执行 finish 方法时,会通过 LoadedApk检查Activity是否存在未注销/解绑的 BroadcastReceiver 和 ServiceConnection。
    如果有未解绑的对象,则通知 AMS 注销/解绑对应的 BroadcastReceiver 和 Service,并打印异常信息,告诉用户应该主动执行注销/解绑的操作。
注:
  • Service的所有生命周期方法和ServiceConnection的回调方法都是运行在主线程的。所以在开发中特别要注意,千万不能在Service的生命周期方法中做非常耗时的操作,否则会引起主线程卡顿,严重时还会引起ANR。

相关文章

网友评论

    本文标题:Android Service调用流程解析

    本文链接:https://www.haomeiwen.com/subject/rkhpbctx.html