Android学习笔记二十六.跨进程调用Service(AIDL Service)

跨进程调用Service(AIDL Service)
一、AIDL Service
1.什么是AIDL Service?
    AIDL,即Android Interface Definition Language.是Android用于定义远程接口,AIDL接口定义语言的语法比较简单,这种接口定义语言并不是真正的编程语言,它只是定义两个进程之间的通信接口。AIDL的语法与Java接口很相似,但存在如下几点差异:
(1)AIDL定义接口的源代码必须以.aidl结尾;
(2)AIDL接口中用到数据类型,除了基本类型、String、List、Map、CharSequence之外,其他类型全部都需要导包,即使他们在同一个包中也需要导包。
    在Android系统中,由于各个应用程序都是运行在自己的进程中(启动一个应用即启动一个进程),进程之间一般无法直接进行数据交换。所以,为了实现进程间的通信(interprocess communiocation,简称IPC),Android提供了AIDL Service。
2.远程Service
    与本地Service相对应的是远程Service,即一个应用程序(进程)可以访问另一个应用程序的Service组件前者我们称之为客户端、后者为服务端,而客户端访问服务端就是通过AIDL接口实现彼此间通信的。通过之前几篇文章我们知道,客户端访问Service时,Android并不是直接返回Service对象给客户端,而是将Service的代理对象(IBinder对象)通过Service的onBind()方法返回给客户端,因此,Android的AIDL远程接口的实现类就是那个IBinder实现类。当客户端获取了远程Service的IBinder对象的代理之后,客户端进程就可通过该IBinder对象去回调远程Service的属性或方法,从而实现进程间的通信。
3.Service返回的IBinder对象
    当访问者(客户端)调用bindService()方法来绑定与启动指定的Service,对于绑定本地Service还是绑定远程Service,Service返回的IBinder对象是有所区别的。
(1)绑定本地Service:本地Service的onBind()方法会直接把IBinder对象本身传给客户端的ServiceConnection对象的onServiceConnected方法的第二个参数;
(2)绑定远程Service:远程Service的onBind()方法只是将IBinder对象的代理传给客户端的ServiceConnection对象的onServiceConnected方法的第二个参数。
4.AIDL接口工作原理
    当在工程中实现了一个.aidl的AIDL接口源文件(如/src/com/android_aidlservice/ICat.aidl),我们可以通过两种方式生成需要的ICat.java接口文件:一是使用Android SDK安装目录下的platform-tools子录下的aidl.exe工具;二是ADT工具自动为该AIDL接口生成实现(路径:gen/com.example.android_aidlservice/ICat.java)。ICat.java接口包含一个Stub内部类,该内部类实现了IBinder、ICat两个接口,这个Stub类将会作为远程Service的回调类---它实现了IBinder接口,因此可作为Service的onBind()方法的返回值
    
二、进程间通信开发思路
1.服务端-远程Service开发
(1)定义一个AIDL接口,文件名为xxx.aidl,保存到源码所在的路径,/src/com/android_aidlservice/ICat.aidl;
package com.example.android_aidlservice;
interface ICat
{
     String getColor();
     double getWeight();
}
其中,com.example.android_aidlservice为其所在的包名。
(2)定义一个Service实现类,并实现一个内部类继承于Stub即实现了ICat接口,也就实现了IBinder接口。该Service的onBind()方法所返回的IBinder对象为ICat.Stub的子类的实例。
private CatBinder catBinder;
public class CatBinder extends Stub
{
    ......
}
@Override
public IBinder onBind(Intent arg0)
{
    return catBinder;
}
(3)在AndroidMainfest.xml工程文件中配置该Service
    <application .... > 
       <!--定义一个Service组件--> 
       <service android:name=".AidlService">
            <intent-filter>
                <action android:name="com.example.sercie.AIDL_SERVICE"></action>
            </intent-filter>
        </service>
  </application>
2.客户端开发思路
(1)定义一个AIDL接口,文件名为xxx.aidl,保存到源码所在的路径,如/src/com/android_aidlservice/ICat.aidl;
package com.example.android_aidlservice;
interface ICat
{
     String getColor();
     double getWeight();
}
其中,com.example.android_aidlservice为其所在的包名,该.aidl文件为从Service段的AIDL接口文件复制到客户端应用中。
(2)创建ServiceConnection对象,并在ServiceConnection对象的onServiceConnected方法中添加如下代码:
    catService = ICat.Stub.asInterface(service);
其中,catService为Service的onBind()方法返回IBinder对象的代理。
(3)以ServiceConnection对象作为参数,调用Context的bindService()方法绑定并启动远程Service即可。
三、源码实现
1.服务端
(1)定义一个Service实现类-/src/com/example/android_service/AidlService.java
package com.example.android_aidlservice;
import java.util.Timer;
import java.util.TimerTask;
import com.example.android_aidlservice.ICat.Stub;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
public class AidlService extends Service {
 //1.声明一个IBinder对象、Timer对象
 private CatBinder catBinder;
 private Timer timer;
 //2.定义两个数组与两个私有变量
 private String color;
 private double weight;
 String[] colors = new String[]{"黑色","黄色","红色"};
 double[] weigths=new double[]{2.3,3.1,1.58};
 //3.继承stub,也就实现了ICat接口,并实现了IBinder接口
 public class CatBinder extends Stub
 {
  @Override
  public String getColor() throws RemoteException {
   return color;
  }
  @Override
  public double getWeight() throws RemoteException {
   return weight;
  }
 
 }
 //4.当Service创建时调用该方法
 @Override
 public void onCreate() {
  super.onCreate();
  //a.实例化一个CatBinder对象
  catBinder = new CatBinder();
  //b.该service完成的任务:随机地改变Service组件内color、weight属性的值
  timer.schedule(new TimerTask(){
   @Override
   public void run() {
    int rand=(int)(Math.random()*3);
    color=colors[rand];
    weight = weigths[rand];
    System.out.println("-----------"+rand);
   }
  }, 0, 800);
 }
 /*5.访问者使用bindService()方法启动Service时,该方法用于返回catBinder对象
  * (1)在绑定本地Service的情况下,该catBinder对象会直接传给客户端的ServiceConnection对象;
  * (2)在绑定远程Service的情况下,只将catBinder对象的代理传递给客户端的ServiceConnection对象的
  * onServiceConnection方法的第二个参数*/
 @Override
 public IBinder onBind(Intent intent) {
  return catBinder;
 }
 //6.回调该方法关闭Service
 @Override
 public void onDestroy() {
  timer.cancel();
 }
}
(2)在AndroidManifest文件中配置该Service
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.android_aidlservice"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="19" />
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <service android:name=".AidlService"
              android:exported="true">
            <intent-filter>
                <action android:name="com.example.service.AIDL_SERVICE"/>
            </intent-filter>
        </service>
    </application>
</manifest>
2.客户端
(1)定义一个Activity实现类-/src/com/example/android_service/AidlClient.java
package com.example.android_aidl_service;
import android.app.Activity;
import android.app.Service;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
public class AidlClient extends Activity {
 private ICat catService;	//声明一个ICat对象
 private Button get;
 private EditText color,weight;
 
 //1.创建一个ServiceConnection对象
 private ServiceConnection conn=new ServiceConnection(){
  @Override
  public void onServiceConnected(ComponentName name, IBinder service) {
   //获取远程Service的onBind方法的对象的代理
   catService = ICat.Stub.asInterface(service);
  }
  @Override
  public void onServiceDisconnected(ComponentName name) {
   catService=null;
  }
 };
 
 //2.绑定远程Service并获取其内容
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  //a.获取Activity主界面组件
  setContentView(R.layout.main);
  get=(Button)findViewById(R.id.get);
  color=(EditText)findViewById(R.id.color);
  weight=(EditText)findViewById(R.id.weight);
 
  //b.创建所需绑定的Service的Intent并设置其Action属性(即指定启动哪个远程Service)
  Intent intent=new Intent();
  intent.setAction("com.example.service.AIDL_SERVICE");
  //c.绑定并启动远程Service
  bindService(intent,conn,Service.BIND_AUTO_CREATE);
  //d.通过IBinder对象的代理获取远程Service数据
  get.setOnClickListener(new OnClickListener(){
   @Override
   public void onClick(View v)
   {
    try
    {
     color.setText(catService.getColor());
     weight.setText(catService.getWeight()+"");
    }
    catch (RemoteException e) {
     e.printStackTrace();
    }
   }
  });
 }
 //3.退出Activity时调用该方法,解除远程Service的绑定
 @Override
 protected void onDestroy() {
  super.onDestroy();
  this.unbindService(conn);
 }

}
(2)在AndroidManifest文件中配置该Activity
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.android_aidl_service"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="19" />
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".AidlClient"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>
效果:运行客户端程序,单击程序界面中"获取远程Service的数据"按钮,就会实现客户端读取远程Service功能。

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。