Service组件 总结 + 绑定理Service三种实现方式 Messager + Binder + AIDL

在Android中进程按优先级可以分为五类,优先级从高到低排列:
- 前台进程 该进程包含正在与用户进行交互的界面组件,比如一个Activity
- 可视进程 该进程中的组件虽然没有和用户交互,但是仍然可以被看到
- 服务进程 该进程包含在执行后台操作的服务组件,比如播放音乐的进程
- 后台进程 该进程包含的组件没有与用户交互,用户也看不到
- 空进程 没有任何界面组件、服务组件,或触发器组件**

Android系统是进程托管的,也就是说进程都是由系统来管理,系统会按照特定的算来来回收这些进程。在回收中秉承几个原则

1. 尽量延长进程的生命周期,不到必须的情况下不会回收,因为系统回收进程会影响用户体验

2. 按优先级从低到高进行回收

3. 同等优先级的进程越近使用越晚回收。


Activity和Service有什么关系
这2者一个前台的东西,一个是后台的东西,Activity有界面,生命周期复杂一点,多个onPause和onResume,Activity不展示的时候并不代表就是关闭了,也有可能是onPause的状态。
Servie没有界面,生命周期也比较简单,但Service分前台服务和后台服务,通过setForeground(boolean);来设置,前台服务比后台服务的存活时间会长。设置不得当服务很容易被系统回收掉。
如果你想让这2个东西有关系,那么请重写Service的onBind方法,这里就是把一个Service和一个Activity绑定的,绑定以后Service就会随着Activity关闭而关闭了。如果Service和Activity没有绑定,那么他俩的生命周期就没有关系。


 

一个常见的应用场景:

        loginactivity登陆页面 点击login, 发送给后台的服务类CmdSocketService 去处理登陆业务,

并获取登陆状态(成功失败的反馈)

loginActivity使用封装好的CmdSocketService(service)类的方法:


a) loginActivity类中定义:


CmdSocketService serviceBinder;


// 下面定义用来连接到服务CmdSocketService后的处理函数
 private ServiceConnection mConn = newServiceConnection()

{
      public void onServiceConnected(ComponentName className, IBinder service)

     {

                   // 这里service其实是MyBinder类对象,,其中有个getService()方法,用于返回service对象自己。

                   // LoginActivity获得了service对象(serviceBinder)的引用,那使用service中方法和普通的类方法一样使用。
              serviceBinder=((CmdSocketService.MyBinder)service).getService();


              Log.v(TAG,"get CmdSocketService 引用 ok !");


              mBound = true;//自己定义的服务是否绑定标记
      }
      public void onServiceDisconnected(ComponentName className) {
              serviceBinder = null;


              mBound = false;
      }  
};


 b) onStart()中定义:

Intent intent = new Intent(this,CmdSocketService.class)


bindService(intent, mConn, Context.BIND_AUTO_CREATE); 


//===========================================================

service类中的定义:

public class CmdSocketService extends Service{

//1 .

public class MyBinder extends Binder

{
      public CmdSocketService getService()

     {


      Log.v(TAG,"getService()");


      return CmdSocketService.this;  //返回service对象本身
      }  
}


// 2.
private MyBinder mBinder = new MyBinder();


//3.反馈给onServiceConnection()

onBind()中实现:

{

       return mBinder;

}


activity和services绑定流程:(bindService方式)

1. new intent指定和哪个service绑定

2. 开始绑定,传递服务连接处理函数ServiceConnection()(有点像回调函数),//调用此函数时android会调用service类中的onBind()函数。

3.onBind()函数里面返回了一个Binder子类对象。Binder子类中有个getServices()方法,返回service对象本身。,最终就是为了给loginActiviy返回service对象的引用。

4.logingActivity和service绑定成功后。android会调用onServiceConnected()函数。此函数中IBinder就是service返回的Binder的子类对象MyBinder.

5.调用MyBinder中的方法getService()即可获得service对象的引用。

6.开始调用service中的公共方法吧。


如何启用Service,如何停用Service

Android中的服务和windows中的服务是类似的东西,服务一般没有用户操作界面,它运行于系统中不容易被用户发觉,可以使用它开发如监控之类的程序。服务的开发比较简单,如下:

第一步:继承Service

public class SMSService extends Service {

}

第二步:在AndroidManifest.xml文件中的<application>节点里对服务进行配置:

<service android:name=".SMSService" />


服务不能自己运行,需要通过调用Context.startService()Context.bindService()方法启动服务。这两个方法都可以启动Service,但是它们的使用场合有所不同。使用startService()方法启用服务,调用者与服务之间没有关连,即使调用者退出了,服务仍然运行。使用bindService()方法启用服务,调用者与服务绑定在了一起,调用者一旦退出,服务也就终止,大有“不求同时生,必须同时死”的特点。


如果打算采用Context.startService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onStart()方法。如果调用startService()方法前服务已经被创建,多次调用startService()方法并不会导致多次创建服务,但会导致多次调用onStart()方法。采用startService()方法启动的服务,只能调用Context.stopService()方法结束服务,服务结束时会调用onDestroy()方法。

 

如果打算采用Context.bindService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onBind()方法。这个时候调用者和服务绑定在一起,调用者退出了,系统就会先调用服务的onUnbind()方法,接着调用onDestroy()方法。如果调用bindService()方法前服务已经被绑定,多次调用bindService()方法并不会导致多次创建服务及绑定(也就是说onCreate()onBind()方法并不会被多次调用)。如果调用者希望与正在绑定的服务解除绑定,可以调用unbindService()方法,调用该方法也会导致系统调用服务的onUnbind()-->onDestroy()方法。

服务常用生命周期回调方法如下:

onCreate() 该方法在服务被创建时调用,该方法只会被调用一次,无论调用多少次startService()bindService()方法,服务也只被创建一次。

onDestroy()该方法在服务被终止时调用。

 

与采用Context.startService()方法启动服务有关的生命周期方法

onStart() 只有采用Context.startService()方法启动服务时才会回调该方法。该方法在服务开始运行时被调用。多次调用startService()方法尽管不会多次创建服务,但onStart() 方法会被多次调用。

 

与采用Context.bindService()方法启动服务有关的生命周期方法

onBind()只有采用Context.bindService()方法启动服务时才会回调该方法。该方法在调用者与服务绑定时被调用,当调用者与服务已经绑定,多次调用Context.bindService()方法并不会导致该方法被多次调用。

onUnbind()只有采用Context.bindService()方法启动服务时才会回调该方法。该方法在调用者与服务解除绑定时被调用

 

采用Context.startService()方法启动服务的代码如下:

public class HelloActivity extends Activity {

    @Override

    public void onCreate(Bundle savedInstanceState) {

        ......

        Button button =(Button) this.findViewById(R.id.button);

        button.setOnClickListener(new View.OnClickListener(){

       public void onClick(View v) {

              Intent intent = new Intent(HelloActivity.this, SMSService.class);

              startService(intent);

       }});       

    }

}

 

采用Context. bindService()方法启动服务的代码如下:

public class HelloActivity extends Activity {

     ServiceConnection conn = new ServiceConnection() {

              public void onServiceConnected(ComponentName name, IBinder service) {

           }

           public void onServiceDisconnected(ComponentName name) {

           }

     };

    @Override public void onCreate(Bundle savedInstanceState) { 

        Button button =(Button) this.findViewById(R.id.button);

        button.setOnClickListener(new View.OnClickListener(){

               public void onClick(View v) {

                  Intent intent = new Intent(HelloActivity.this, SMSService.class);

                  bindService(intent, conn, Context.BIND_AUTO_CREATE);

                  //unbindService(conn);//解除绑定

          }});       

    }

}


 

绑定Service的三种实现方式之使用Messeager  1

如果你需要在不同进程间通信,你可以在Service中使用Messenger来实现进程中通信。

如果使用这种方式,Service中需要定义一个Handler对象(负责对客户端发送过来的Message进行响应)。

Messenger可以共享给client一个IBinder对象,client通过这个IBinder对象向Service发送Message,而前面提到的Handler对象是这一切的基础。

注:使用这种方式进行通信是不支持多线程的。

那就让我们来看看使用这种方式进行通信吧!

注:Service在声明时必须对外开放,即android:exported="true",且本文是通过Intent启动的Service,所以在声明时该Service可以接收特定的Action。

1、在Service中创建一个Handler对象,来处理从client发过来的Message

2、根据创建的Handler对象创建一个Messenger对象

3、使用Messenger的getBinder方法得到一个IBinder对象,并在Service的onBind方法中将其反出去

4、client在onServiceConnected中根据IBinder参数创建一个Messenger对象(可参考Messenger的构造函数)

5、client可以使用上一步得到的Messenger对象来给Service发送Message了

经过上面的五部我们就能让client与Service进行通信。client使用Messenger对象给Service发送Message后,Service中的Handler将会对消息作出响应。

上面实现的仅仅是单向通信,即client给Service发送消息,如果我需要Service给client发送消息又该怎样做呢?

其实,这也是很容易实现的,下面就让我们接着上面的步骤来实现双向通信吧~

6、在client中创建一个Handler对象,用于处理Service发过来的消息

7、根据client中的Handler对象创建一个client自己的Messenger对象

8、我们在第5步的时候获得了Service的Messenger对象,并通过它来给Service发送消息。这时候,我们将client的Messenger对象赋给待发送的Message对象的replyTo字段

9、在Service的Handler处理Message时将client的Messenger解析出来,并使用client的Messenger对象给client发送消息

这样我们就实现了client和Service的双向通信。client和Service都有自己的Handler和Messenger对象,使得对方可以给自己发送消息,值得注意的是client的Messenger是通过Message的replyTo传递给Service的。

Demo链接:http://pan.baidu.com/share/link?shareid=583593&uk=2953765628


 

绑定Service的三种实现方式之继承Binder类  2

继承Binder类实现绑定Service的应用场合:Service仅供自己使用(不对第三方程序开发)。

注:这种方式仅适用于client和service在同一个程序和进程的情况。

实现方法:

1、在Service中创建一个Binder的实例:

    这个实例包含client可以调用的公共方法;

    这个实例返回当前Service对象(该Service实例包含client可以调用的公共方法)

    这个实例返回Service类中的一个类对象,而这个类对象包含client可以调用的公共方法

2、在Service的onBind函数中返回这个Binder实例

3、在client端的onServiceConnected方法中获得这个Binder实例,并通过这个Binder实例调用Service端的公共方法。

Demo请转至:http://pan.baidu.com/share/link?shareid=582094&uk=2953765628


 

绑定Service的三种实现方式之使用AIDL  3

AIDL全称为Android Interface Definition Language,它可以使你的程序实现进程间通信(IPC),并且在实现IPC的基础上允许多线程访问。

首先,我们要创建一个自己的.aidl文件(见Demo中的IRemoteService.aidl)。

定义AIDL文件与java中创建接口非常的类似。一般来说,aidl支持的数据类型有五种:java基本数据类型;String;CharSequence;List;Map。其中List和Map较为特殊(http://developer.android.com/guide/components/aidl.html#Create)。如果你使用的数据类型不是AIDL的基本数据类型,你必须要使用import语句将其导入,即使他们是在同一个package下。声明方法时,方法的参数可以零到多个,返回值可以是void;所有的非基本数据类型都需要指定是传入还是传出值(基本数据类型都是传入值;在AIDL文件中不能声明静态字段)

其次,我们要实现我们上面刚刚创建的接口(见Demo中StudentService中的mBinder)。

假设我们有一个AIDL文件叫IRemoteService.aidl,当我们编译我们的项目的时候,android的ant能将我们的AIDL文件生成为java文件(放在gen/下面)。这个java文件中有一个抽象内部类Stub(继承了Binder类)实现了我们的接口,并提供了一个asInterface方法将IBinder对象转化为我们的接口类型。因为实现我们的接口就转化为实现其抽象内部类Stub(Service端的业务函数全部在这里实现)。

再次,将我们的接口暴露给客户端(注意Demo在Server端AndroidManifest.xml中对StudentService的声明)

将我们的接口暴露给客户端实际上就是:在Service的onBind函数中将我们的Stub类的实例反出去。

这样,Service端的工作就完成了!

那么,怎样通过IPC传递对象呢(见Demo中的Student.java和Student.aidl)?

如果你想在进程间传递对象,那么对象就必须实现Parcelable接口。而实现这个接口需要我们完成以下几步:

1、在声明对象的时候实现Parcelable接口

2、实现writeToPacel方法

3、添加一个叫做CREATOR的静态变量(这个变量要实现Parcelable.Creator接口)

4、创建一个AIDL文件来声明这个实现了Parcelable接口的类

这样我们就可以在AIDL中使用对象了。

客户端怎么对Service端的方法进行调用呢?

其实很简单,只需要一下几步就可以了(见Demo中的Client中的代码)。

1、将Server端使用的.aidl文件拷贝到client程序中,如果你的aidl文件仅仅是对实现了Parcelable接口的类的说明,那么,对应的java文件也要拷贝过去。(注意包名,具体可参考Demo的client端对Sever端AIDL文件的导入)

2、编译client程序,将在gen/目录下生成AIDL对应的的类文件

3、实现ServiceConnection接口。在onServiceConnected方法中,使用AIDL生成的Java文件的函数(内部类Stub的asInterface)将onServiceConnected函数的中IBinder参数转化为AIDL生成的接口对象。

4、根据第3步得到的对象来调用AIDL中的函数。

5、调用bindService函数执行绑定操作(解绑使用unbindService函数)

到这里,AIDL的使用讲解就结束了。推荐你结合本文的Demo来读这篇文章,相信会让你受益匪浅的。

Demo链接:http://pan.baidu.com/share/link?shareid=587849&uk=2953765628

 

 


你可能感兴趣的:(service)