AndroidService 深度解析(1)

来源:互联网 发布:c语言计算成绩平均分 编辑:程序博客网 时间:2024/06/06 02:13

AndroidService 深度解析(1)

      这段时间我重新对Android基础进行了学习,还是先从官方文档开始,做了关于Service的文档翻译,熟悉了关于Service的知识点。但是在用的时候还是有很多需要注意的地方,需要把这些知识点联系起来,因此有了这篇文章。我将主要从以下几个方面对Service类进行深度解析:

1、 Service的生命周期(看起来简单,实际上很复杂的);

2、 绑定运行Service(Bound Service)的多种形式实现及其使用;

      主要包括本地Service继承Binder类的实现、远程Service使用Messenger实现、远程Service的AIDL方式实现。(本来打算这篇一起写,篇幅太长,作为两篇吧)

      如果您有兴趣继续阅读下去,但是您对Service的基础还不怎么熟悉的话,建议您先阅读以下三篇我对官方文档的翻译:

Service API文档 翻译:

http://blog.csdn.net/chtsx/article/details/44276105

Services 综合介绍 :

http://blog.csdn.net/chtsx/article/details/44276189

Bound Service 以绑定方式运行的Service :

http://blog.csdn.net/chtsx/article/details/44276241


Service的生命周期


       虽然Service在它的生命周期中涉及到的方法一共只有onCreate、onStartCommand、onBind、onUnbind、onDestory这几个,但是由于Service拥有两种运行方式,而且可以实现同时为多个客户端服务,支持跨进程,使得它的运行时状态真的很难让人清楚地掌握。而官方给的这个图也很难将Service的生命周期详细地表述。


       为了好好把握Service在不同状态下的运行,我做了多种可能情况下的测试:

       在Android Studio下建立项目AndroidServiceFULLStudy,并新建一个名称为ServiceTest的工程。在ServiceTest工程下,建立一个ServiceLife类、一个MainActivity类,MainActivity与ServiceLife类中将我们执行的操作,与系统调用的生命周期方法都以日志的形式打印出来便于分析。


ServiceLife类:

public class ServiceLife extends Service {    private String TAG = getClass().getSimpleName();    MyBinder myBinder=new MyBinder();    public ServiceLife() {    }    public class MyBinder extends Binder {        public ServiceLife getService() {            return ServiceLife.this;        }    }    @Override    public void onCreate() {        Log.d(TAG, "ServiceLife------>onCreate()");        super.onCreate();    }    @Override    public int onStartCommand(Intent intent, int flags, int startId) {        Log.d(TAG, "ServiceLife------>onStartCommand()");        return super.onStartCommand(intent, flags, startId);    }    @Override    public IBinder onBind(Intent intent) {        Log.d(TAG, "ServiceLife------>onBind()");        return myBinder;    }    @Override    public void onRebind(Intent intent) {        Log.d(TAG, "ServiceLife------>onRebind()");        super.onRebind(intent);    }    @Override    public boolean onUnbind(Intent intent) {        Log.d(TAG, "ServiceLife------>onUnbind()");        //return false;        return true;    }    @Override    public void onDestroy() {        Log.d(TAG, "ServiceLife------>onDestroy()");        super.onDestroy();    }}


MainActivity类:       

<pre name="code" class="java">public class MainActivity extendsActionBarActivity {   private String TAG = getClass().getSimpleName();   Intent serviceIntent;   @Override   protected void onCreate(Bundle savedInstanceState) {     super.onCreate(savedInstanceState);     setContentView(R.layout.activity_main);     Log.d(TAG,"MainActivity------>onCreate()");     serviceIntent = newIntent(this,ServiceLife.class);}    publicvoid startService(View v) {     Log.d(TAG,"MainActivity------>startService()");     startService(serviceIntent);    }   public void stopService(View v) {     Log.d(TAG,"MainActivity------>stopService()");     stopService(serviceIntent);    }   public void bindService(View v) {     Log.d(TAG,"MainActivity------>bindService()");     bindService(serviceIntent, serviceConnection,Service.BIND_AUTO_CREATE);    }   public void unBindService(View v) {     Log.d(TAG, "MainActivity------>unBindService()");     unbindService(serviceConnection);    }   @Override   protected void onDestroy() {     Log.d(TAG,"MainActivity------>onDestroy()");     super.onDestroy();    }   ServiceConnection serviceConnection = new ServiceConnection() {     @Override     public void onServiceConnected(ComponentNamename, IBinder service) {      Log.d(TAG,"MainActivity------>onServiceConnected()");      ServiceLife.MyBinderbinder = (ServiceLife.MyBinder) service;      ServiceLifeserviceLife = binder.getService();     }     @Override     public voidonServiceDisconnected(ComponentName name) {      Log.d(TAG,"MainActivity------>onServiceDisconnected()");     }    };}


 MainActivity的界面布局这里就不再附上代码,贴上截图:


       图中四个按钮分别对应执行MainActivity中startService、stopService、bindService、unBindService四个方法。

      我们测试的过程如下,然后通过打印的日志进行分析:

1多次点击STARTSERVICE按钮,再多次点击STOPSERVICE按钮:

03-1516:55:26.924    D/MainActivity﹕MainActivity------>onCreate()

03-15 16:55:30.554   D/MainActivityMainActivity------>startService()

03-1516:55:30.574    D/ServiceLife﹕ServiceLife------>onCreate()

03-1516:55:30.574    D/ServiceLife﹕ServiceLife------>onStartCommand()

03-15 16:55:39.644   D/MainActivityMainActivity------>startService()

03-1516:55:39.644    D/ServiceLife﹕ ServiceLife------>onStartCommand()

03-15 16:55:40.024   D/MainActivityMainActivity------>startService()

03-1516:55:40.034    D/ServiceLife﹕ServiceLife------>onStartCommand()

03-15 16:55:40.304   D/MainActivityMainActivity------>startService()

03-1516:55:40.314    D/ServiceLife﹕ServiceLife------>onStartCommand()

03-15 16:55:45.144   D/MainActivityMainActivity------>stopService()

03-1516:55:45.144    D/ServiceLife﹕ServiceLife------>onDestroy()

03-15 16:55:45.504   D/MainActivity MainActivity------>stopService()

03-15 16:55:45.824   D/MainActivityMainActivity------>stopService()

03-15 16:55:46.054   D/MainActivityMainActivity------>stopService()

03-1516:55:56.114    D/MainActivity﹕MainActivity------>onDestroy()

       如上日志显示,第一次点击STARTSERVICE时,将执行ServiceLife的onCreate与onStartCommand方法,此后,由于Service已经构建运行,之后点击STARTSERVICE按钮时,就只会执行onStartCommand方法;无论我们点击多少次STARTSERVICE按钮,我们只需点击一次STOPSERVICE按钮,Service就会执行onDestory销毁,之后我们继续点击STOPSERVICE按钮将不会有任何作用。


2多次点击BINDSERVICE按钮,再多次点击UNBINDSERVICE按钮:

03-15 17:04:04.514   D/MainActivity﹕MainActivity------>onCreate()

03-15 17:04:08.354  D/MainActivityMainActivity------>bindService()

03-15 17:04:08.374   D/ServiceLife﹕ServiceLife------>onCreate()

03-15 17:04:08.374   D/ServiceLife﹕ ServiceLife------>onBind()

03-15 17:04:08.384   D/MainActivity﹕MainActivity------>onServiceConnected()

03-15 17:04:09.174  D/MainActivityMainActivity------>bindService()

03-15 17:04:09.664  D/MainActivityMainActivity------>bindService()

03-15 17:04:09.954  D/MainActivityMainActivity------>bindService()

03-15 17:04:10.274  D/MainActivityMainActivity------>bindService()

03-15 17:04:13.824  D/MainActivityMainActivity------>unBindService()

03-15 17:04:13.824   D/ServiceLife﹕ServiceLife------>onUnbind()

03-15 17:04:13.824   D/ServiceLife﹕ServiceLife------>onDestroy()

03-15 17:04:16.944  D/MainActivityMainActivity------>unBindService()

03-1517:04:16.944   W/dalvikvm﹕ threadid=1:thread exiting with uncaught exception

       如上日志显示,第一次点击BINDSERVICE按钮时,由于Service尚未运行,ServiceLife将会执行onCreate与onBind方法,当ServiceLife与MainActivity真正建立连接时,onServiceConnected将会被调用,但是之后多次点击BINDSERVICE按钮就没有意义了,当我们第一次点击UNBINDSERVICE时,ServiceLife将会执行onUnbind与onDestory方法,证明Service已经被销毁,当我们之后再点击UNBINDSERVICE时,程序将会发生运行时异常。因此,当并未与Service连接时,我们进行解绑,将会发生异常,这在编程中要注意。


3按以下顺序依次点击:STARTSERVICE—>BINDSERVICE—>UNBINDSERVICE—> STOPSERVICE

03-1517:18:58.774     D/MainActivity MainActivity------>startService()

03-1517:18:58.794     D/ServiceLife﹕ServiceLife------>onCreate()

03-1517:18:58.794     D/ServiceLife﹕ServiceLife------>onStartCommand()

03-1517:19:01.014     D/MainActivity MainActivity------>bindService()

03-1517:19:01.024     D/ServiceLife﹕ServiceLife------>onBind()

03-1517:19:01.034     D/MainActivity﹕MainActivity------>onServiceConnected()

03-1517:19:02.334     D/MainActivity MainActivity------>unBindService()

03-1517:19:02.334     D/ServiceLife﹕ServiceLife------>onUnbind()

03-1517:19:06.364     D/MainActivity MainActivity------>stopService()

03-1517:19:06.364     D/ServiceLife﹕ServiceLife------>onDestroy()

       如上日志显示,点击STARTSERVICE按钮时,由于Service尚未运行,所以首先执行onCreate方法,再执行onStartCommand方法,之后再点击BINDSERVICE按钮时,由于Service已经运行,所以将会直接执行onBind方法,连接建立时,onServiceConnected得到调用;当我们点击UNBINDSERVICE时,onUnbind方法将得到调用,但是Service并不会被销毁;之后点击STOPSERVICE按钮,Service将会执行onDestory销毁。   当然,这种情况是最正常的情况,我们应该都能做出正常判断,但是,以下情况您是否能清楚地判断呢?


4按以下顺序依次点击:STARTSERVICE—>BINDSERVICE—> STOPSERVICE—> UNBINDSERVICE

       看到这里你会不会觉得我已经调用stopService方法,这时Service会被销毁吧,,后面又去调用UNBINDSERVICE?会发生异常吗?我们来看打出的日志:

03-1517:29:02.164     D/MainActivity﹕MainActivity------>onCreate()

03-1517:29:13.214     D/MainActivity MainActivity------>startService()

03-1517:29:13.234     D/ServiceLife﹕ServiceLife------>onCreate()

03-15 17:29:13.254     D/ServiceLife﹕ServiceLife------>onStartCommand()

03-1517:29:14.294     D/MainActivity MainActivity------>bindService()

03-1517:29:14.314     D/ServiceLife﹕ServiceLife------>onBind()

03-1517:29:14.324     D/MainActivity﹕MainActivity------>onServiceConnected()

03-1517:29:15.354     D/MainActivity MainActivity------>stopService()

03-1517:29:17.674     D/MainActivity MainActivity------>unBindService()

03-1517:29:17.674     D/ServiceLife﹕ServiceLife------>onUnbind()

03-1517:29:17.674     D/ServiceLife﹕ServiceLife------>onDestroy()

       看到上面的日志,你会不会惊讶地发现在调用stopService之后,Service竟然无动于衷?当后面点击UNBINDSERVICE的时候,Service才解绑并销毁?这里我们先不给出结论,看下一个测试。


5按以下次序依次点击:BINDSERVICE—> STARTSERVICE—>UNBINDSERVICE—> STOPSERVICE

       这次你又会怎样猜测呢?居然先绑定,又去调用startService方法?这不乱扯吗?哈哈。看看下面的日志:

03-15 17:40:52.914     D/MainActivity﹕MainActivity------>onCreate()

03-15 17:40:57.334    D/MainActivityMainActivity------>bindService()

03-15 17:40:57.364     D/ServiceLife﹕ServiceLife------>onCreate()

03-15 17:40:57.364     D/ServiceLife﹕ServiceLife------>onBind()

03-15 17:40:57.374     D/MainActivity﹕MainActivity------>onServiceConnected()

03-15 17:40:57.964    D/MainActivityMainActivity------>startService()

03-15 17:40:57.974     D/ServiceLife﹕ ServiceLife------>onStartCommand()

03-15 17:40:58.714    D/MainActivityMainActivity------>unBindService()

03-15 17:40:58.724     D/ServiceLife﹕ServiceLife------>onUnbind()

03-15 17:40:59.274    D/MainActivityMainActivity------>stopService()

03-15 17:40:59.274     D/ServiceLife﹕ServiceLife------>onDestroy()

       看到如上日志,你是恍然大悟还是依然摸不着头脑呢?当我们点击BINDSERVICE按钮时,执行Service的onCreate、onBind方法,然后连接真正建立时调用onServiceConnected方法,这些都没问题,然后我们点击STARTSERVICE按钮,由于Service已经启动,所以直接执行onStartCommand方法,然后我们点击UNBINDSERVICE按钮,onUnbind方法被调用,但是并未销毁Service,直到点击STOPSERVICE按钮,onDestory才会执行。   现在你是否有些头绪了呢?如果没懂,别急,我们再进行一个测试以后,我就把总的结论写出来。


6按以下次序依次点击:STARTSERVICE—>BINDSERVICE—> UNBINDSERVICE—>BINDSERVICE—> UNBINDSERVICE—>STOPSERVICE

03-15 17:54:16.404     D/MainActivity﹕MainActivity------>onCreate()

03-15 17:54:20.414    D/MainActivityMainActivity------>startService()

03-15 17:54:20.434     D/ServiceLife﹕ServiceLife------>onCreate()

03-15 17:54:20.434     D/ServiceLife﹕ServiceLife------>onStartCommand()

03-15 17:54:21.364    D/MainActivityMainActivity------>bindService()

03-15 17:54:21.374     D/ServiceLife﹕ServiceLife------>onBind()

03-15 17:54:21.384     D/MainActivity﹕MainActivity------>onServiceConnected()

03-15 17:54:22.074    D/MainActivityMainActivity------>unBindService()

03-15 17:54:22.074     D/ServiceLife﹕ServiceLife------>onUnbind()

03-15 17:54:26.224    D/MainActivityMainActivity------>bindService()

03-15 17:54:26.234     D/MainActivity MainActivity------>onServiceConnected()

03-15 17:54:26.234     D/ServiceLife﹕ServiceLife------>onRebind()

03-15 17:54:27.214    D/MainActivityMainActivity------>unBindService()

03-15 17:54:27.214     D/ServiceLife﹕ServiceLife------>onUnbind()

03-15 17:54:37.144     D/MainActivity﹕MainActivity------>stopService()

03-15 17:54:37.154     D/ServiceLife﹕ServiceLife------>onDestroy()

       哈哈哈,看到上面这个日志,真是让我哭笑不得!居然第二次绑定的时候直接就调用了onServiceConnected方法,之后又出现了个onRebind方法。

       不知道细心的你在看我贴出的代码时有没有发现在ServiceLife类中我将onUnbind方法改为永远返回true了呢?那么把它改成true有什么作用呢?这里我贴一张官方的图,能很好地解释。我认为这张图逻辑表述得不错,就是丑了点,哈哈。


       算了,为了更加有力地证明上图的正确性(官方都给图了,能错?原谅我的强迫症),与我下面的猜测,我将ServiceLife类中的onUnbind返回false,再把日志打印下。

03-1518:08:28.134       D/MainActivity﹕MainActivity------>onCreate()

03-1518:08:31.244       D/MainActivity MainActivity------>startService()

03-1518:08:31.264       D/ServiceLife﹕ServiceLife------>onCreate()

03-1518:08:31.264       D/ServiceLife﹕ServiceLife------>onStartCommand()

03-1518:08:34.344       D/MainActivity MainActivity------>bindService()

03-1518:08:34.354       D/ServiceLife﹕ServiceLife------>onBind()

03-1518:08:34.364       D/MainActivity﹕MainActivity------>onServiceConnected()

03-1518:08:37.524       D/MainActivity MainActivity------>unBindService()

03-1518:08:37.524       D/ServiceLife﹕ServiceLife------>onUnbind()

03-1518:08:39.204       D/MainActivity MainActivity------>bindService()

03-15 18:08:39.214      D/MainActivity MainActivity------>onServiceConnected()

03-1518:08:43.364       D/MainActivity MainActivity------>unBindService()

03-1518:08:45.274       D/MainActivity﹕MainActivity------>stopService()

03-1518:08:45.284       D/ServiceLife﹕ServiceLife------>onDestroy()

       看到这个日志我真的想哭,我经过反复测试,结果就是这样!我对天发誓,在做上一个实验的时候,我都没想到结果会是这样!说好的官方的呢?第二次绑定的时候,onBind方法去哪里了呢?

       我使用的是小米1青春版,系统是最新开发版,基于ANDROID 4.1.2的,看到这里的朋友说明你的耐心真的很不错,有兴趣的话在你的机子上测试下会不会执行onBind方法,具体工程后面我会贴出链接。

------>      其实具体第二次绑定时onBind是否执行也没多大关系,一般我们也不会在里面去处理其他逻辑。如果有在onBind中处理逻辑的朋友注意下就是了。

 

       好了,上面的测试现在告一段落,Service的生命周期确实会很复杂,官方给出的生命周期图也很模糊,当然我也很难画一个很好的图,但是我可以从几个关键方法的角度,进行比较准确的解释。

onCreate()

      无论我们使用startService还是bindService方法启动Service,只要当前Service没有运行,就会执行onCreate方法;但是如果这个Service已经在执行了,那么onCreate方法就不会执行。记住,onCreate与onDestory成对出现;

onDestory():

      简单地说,就是Service销毁的时候执行。但是具体什么时候,很难描述,不好写,下面做一个表述。

      我们认为,只要调用过startService,那么Service就以独立方式运行;只要调用过bindService,那么Service就以绑定方式运行;如果startService和bindService都调用过,那么Service既是独立运行的又是绑定运行的。

      对于以独立方式运行的Service,只要我们调用stopService,那么它就退出独立运行方式,无论stopService之前调用过多少次startService,都会使Service退出独立运行方式;

      对于以绑定方式运行的Service,只要我们调用unbindService,那么就表示我这个客户端取消了与Service的绑定,而不论我这个客户端之前使用过多少次bindService,只需要一次unbindService就取消了绑定,而且只有绑定的客户端才能使用unbindService方法,未绑定客户端使用这个方法将会爆出异常;那么当所有的客户端都与Service取消绑定时,Service就退出了绑定运行方式。ServiceonBind方法执行,表示Service进入绑定模式,ServiceonUnbind方法执行表示Service退出绑定模式。但是,onBind方法不执行,Service也可能进入绑定模式,就像上面第56个测试一样。

      那么最后得出结论:当Service既不是已独立方式运行,又不是以绑定方式运行,这个Service就会被销毁了,onDestory就会执行。

onStartCommand(Intent intent, int flags, int startId):

      这个方法的执行时机比较简单,每次调用startService都会执行onStartCommand方法。参数Intent就是每次startService传入的intent,我们可以通过intent携带的数据,识别我们具体需要执行的任务。

onBind(Intent intent)

      当Service的第一个客户端第一次通过bindService方法绑定到这个Service的时候,才会执行onBind方法,后续的绑定都不再执行这个方法。它返回一个IBinder对象,这个IBinder对象将交给系统管理,后续调用bindService的时候,系统直接将IBinder对象返回给客户端,而不再调用onBind方法。

      官方文档说,当Service运行于绑定模式和独立运行模式时,如果退出绑定模式时onUnbind返回了false,那么当再次进入绑定模式时将会执行onBind方法,这在我的测试中不是这样的,因此需要注意。

onUnbind(Intent intent):

      当所有与Service绑定的客户端全部解绑时,即Service退出绑定模式时,执行。

onRebind(Intent intent)

      当Service运行于绑定模式和独立运行模式时,如果退出绑定模式时onUnbind返回了true,那么当再次进入绑定模式时将会执行onRebind方法,而不会执行onBind方法。

 

 

       好了,这篇文章对Service生命周期各个方法做了测试和总结,相信你也有所收获吧。下一篇我将对如何实现Bound Service做具体介绍,有兴趣的可以继续看。

       本来准备花一个下午把Service相关问题弄清楚,结果一开始弄就蒙了,还是挺复杂的,花了三四天了,不过现在还是比较清楚。   当然,Service还有很多注意事项,比如处理系统杀死Service再重启的处理,这个在以后具体开发的时候再总结吧。

       本篇文章涉及到的代码见Github仓库:https://github.com/BBigBoy/AndroidServiceFULLStudy

 

0 0
原创粉丝点击