这两天在学习aidl跨进程调用,研究了一天终于写出了个helloword。。做下记录吧。
根据官方提供的指南,开发aidl跨进程调用,主要分以下步骤:
1.创建.aidl文件-该文件(YourInterface.aidl)定义了客户端可用的方法和数据的接口。
2.在makefile文件中加入.aidl文件-(Eclipse中的ADT插件提供管理功能)Android包括名为AIDL的编译器,位于tools/文件夹。
3.实现接口-AIDL编译器从AIDL接口文件中利用Java语言创建接口,该接口有一个继承的命名为Stub的内部抽象类(并且实现了一些IPC调用的附加方法),要做的就是创建一个继承YourInterface.Stub的类并且实现在.aidl文件中声明的方法。
4.向客户端公开接口-如果是编写服务,应该继承Service并且重载Service.onBind(Intent) 以返回实现了接口的对象实例
上面的东西真的很官方。。下面结合我的例子做一下学习总结:
先开发服务器端
1.创建一个aidl文件,可以看提供的源码。创建完成后,android studio构建工具会自动在module/build/generated/source/aidl/debuge/pakagename/目录下生成同名字的java文件。里面包含一个Stub抽象类,这个类继承自android.os.Binder,这个类是实现整个远程调用的核心。
2.然后创建一个类来继承上面说到的那个Stub抽象类,实现里面的抽象方法。(这些抽象方法是根据aidl文件自动生成的)。
3.创建一个自定义Service继承自Service,实现其onBind方法,注意此onBind方法必须返回第二步创建的那个Stub类的子类。然后在xml中声明此service,注意此service的声明必须包含一个action,此action也用于客户端的调用使用。(在下面的客户端开发中会有介绍)。
4.创建一个activity,此activity只要实现把service启动了即可。
这样服务器端就开发完毕,运行后启动了一个可供远程调用的service。关键还是通过onBind暴露一个Binder给客户端。Binder哪来呢?就是通过aidl文件adt会自动生成一个抽象类Stub继承自Binder,只需要创建一个类实现这个Stub的抽象方法即可。
然后开发客户端:
1.客户端也需要一个aidl文件,注意客户端的aidl文件的包名必须和服务器端的aidl包名一致,名字也相同。所以可以直接将服务端的拷贝到客户端。
2.创建一个Activity,包含变量ServiceConnection con,实现其onServiceConnected和onServiceDisconnected方法,onServiceConnected方法生成第一步那个接口的实现类的对象。con对象用于在onCreate中绑定service,这个service的action必须为服务器端声明的那个service的配置action。绑定中用到con会执行onServiceConnected方法生成aidl对象iPerson。然后就可以通过iPerson来调用aidl里的任意方法返回服务器的东西。
客户端开发完毕。关键点是创建aidl文件自动生成了一个接口,在activity中必须绑定服务程序开启的service,在绑定过程中初始化aidl对象。然后就可用aidl对象调用任意方法了。
OK,过程真的很繁琐。。但是好像也只能这样了。最主要的还是服务端onBind暴露,然后客户端bindService得到aidl对象。
接下来是在Android Studio环境下写的AIDL小例子,实现步骤如下:
1.创建服务端
1.1 创建一个module,名称为aidlserverdemo,然后在main目录下创建一个aidl文件IPerson.aidl,创建成功后会自动在目录module/build/generated/source/aidl/debuge/pakagename/xxx.java下生成.java的文件,如果没有自动生成,rebuild一下之后一般会生成.
IPerosn.aidl
package com.sg7.aidlserverdemo;
interface IPerson {
void setAge(int age);
void setName(String name);
String display();
}
生成的.java文件:
1.2 创建IPersonImpl类,继承IPerson的子类IPerson.Stub
public class IPersonImpl extends IPerson.Stub {
// 声明两个变量
private int age;
private String name;
@Override
public void setAge(int age) throws RemoteException {
this.age = age;
}
@Override
public void setName(String name) throws RemoteException {
this.name = name;
}
@Override
public String display() throws RemoteException {
return "name:zzz;age=18";
}
}
1.3 创建一个远程服务MyRemoteService,在AndroidManifest.xml中声明action为com.sg7.app.action.MY_REMOTE_SERVICE
/**
* @author zzz
* 使用Service将接口暴露给客户端
*/
public class MyRemoteService extends Service {
// 声明IPerson接口
private Binder iPerson = new IPersonImpl();
@Override
public IBinder onBind(Intent intent) {
return iPerson;
}
}
<service android:name=".MyRemoteService">
<intent-filter>
<action android:name="com.sg7.app.action.MY_REMOTE_SERVICE"/>
</intent-filter>
</service>
1.4 在MainActivity中启动服务
public class MainActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent();
// 设置Intent的Action 属性
intent.setAction("com.sg7.app.action.MY_REMOTE_SERVICE");
// 绑定服务
startService(intent);
}
}
整个项目结构
2.创建客户端
2.1 创建一个module,名称为aidlclientdemo,然后将aidlserverdemo中的aidl整个文件夹复制到aidlclientdemo的main文件夹下,复制成功后会自动在目录module/build/generated/source/aidl/debuge/pakagename/xxx.java下生成.java的文件,如果没有自动生成,rebuild一下之后一般会生成.
2.2 在MainActivity中通过ServiceConnection获得IPerson接口,并且调用其中的方法
public class MainActivity extends Activity {
// 声明IPerson接口
private IPerson iPerson;
private Button btn;
// 实例化ServiceConnection
private ServiceConnection conn = new ServiceConnection() {
@Override
synchronized public void onServiceConnected(ComponentName name, IBinder service) {
// 获得IPerson接口
iPerson = IPerson.Stub.asInterface(service);
System.out.println("iperson----------:" + iPerson);
}
@Override
public void onServiceDisconnected(ComponentName name) {
iPerson = null;
}
};
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btn = (Button) findViewById(R.id.Button01);
Intent intent = new Intent("com.sg7.app.action.MY_REMOTE_SERVICE");
bindService(intent, conn, Service.BIND_AUTO_CREATE);
btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
try {
String msg = iPerson.display();
// 显示方法调用返回值
Toast.makeText(MainActivity.this, msg, Toast.LENGTH_LONG).show();
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
@Override
protected void onDestroy() {
unbindService(conn);
super.onDestroy();
}
}
效果图: