AIDL是Android跨进程间的一种非常重要的IPC通信机制,今天我们将来详细讲述如何不同app之间如何进行AIDL IPC通信。
前期基础知识储备
-
IPC定义:
IPC是intent-Process Communication的缩写,含义为进程间通信或者跨进程通信,是指两个进程之间进行数据交换的过程。IPC不是Android所独有的,任何一个操作系统都需要有相应的IPC机制,比如Windows上可以通过剪贴板、管道和邮槽等来进行进程间通信,而Linux上可以通过命名共享内容、信号量等来进行线程间通信。对于Android来说,它也有自己的进程间通信方式,Android建构在Linux基础上,继承了一部分Linux的通信方式,同时Android也有自己特有的通信方式。
创建应用工程
-
创建AIDL Service端工程model
APP
-
创建AIDL文件
右击Java目录 New --> AIDL --> AIDL File
输入接口名称IMyAidlInterface
此时会在与java同一目录下生成aidl文件夹及对应的包名和aidl文件
AS自动帮我们生成了一个基本类型的aidl接口
interface IMyAidlInterface{
/**
* Demonstrates some basic types that you can use as parameters
* and return values in AIDL.
*/
void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
double aDouble, String aString);
}
但我们并不打算用它,我们自己来定义一个简单的基本类型IOnroad
interface IOnroad {
/**
* Demonstrates some basic types that you can use as parameters
* and return values in AIDL.
*/
// void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
// double aDouble, String aString);
void sayHello(String name,int age);
}
点击菜单Build-->Make Project,即可在对应目录下生成Java接口文件,如图所示:
-
创建一个service来实现这个接口类
我们创建了一个OnroadService, 并在onBind方法里返回mBinder给调用者,调用者就可以透过mBinder调用我们的aidl接口了。
public class OnroadService extends Service {
public OnroadService() {
}
private String TAG = "OnroadService";
private IOnroad.Stub binder = new IOnroad.Stub() {
@Override
public void sayHello(String name, int age) throws RemoteException {
String text = "sayHello: --->name" + name + "age:" + age;
Log.i(TAG, text);
//Toast.makeText(OnroadService.this, text, Toast.LENGTH_SHORT).show();
}
};
@Override
public IBinder onBind(Intent intent) {
return binder;
}
}
-
在AndroidManifest.xml注册该service
<service
android:name=".service.OnroadService"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.lypeer.aidl" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</service>
创建Client端应用
-
创建AIDL Client端工程工程model
-
复制service端的aidl文件及aidl对应的java文件到客户端
为了能在Client端掉用service端的接口,必须将service端的aidl文件及AS自动生成的aidl对应的java文件复制到该工程,且保持包名不变。
-
创建ServiceConnection类
用于绑定Service, 若绑定成功,会回调onServiceConnected()方法,我们可以复写该方法,获取该service的ibinder接口; 若Service断开连接,则必须销毁binder实例,否则会造成内存泄漏。
private ServiceConnection connection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// IOnroad.Stub binder =
IOnroad onroad = IOnroad.Stub.asInterface(service);
try {
// onroad.sayHello();
onroad.sayHello("哈喽",100);
} catch (RemoteException e) {
e.printStackTrace();
Toast.makeText(MainActivity.this, "有异常的情况!", Toast.LENGTH_SHORT).show();
}
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
-
绑定服务
为了方便,我们让Activity启动的时候就去绑定服务
setAction里的参数tech.onroad.aidlserverdemo就是我们Service在AndroidManifest.xml注册的action, setPackage需指定service所在的包名,因为在Android 5.0以后,android已不再支持隐式Intent Service,所以必须指定service所在的包名。这样Client端绑定Service就已经完成。
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent();
intent.setAction("com.lypeer.aidl");
intent.setPackage("com.zxn.service");
bindService(intent,connection,BIND_AUTO_CREATE);
}
-
调用aidl接口
接下来我们尝试调用一下service端的aidl接口,看能否调用成功。
测试
可以看到服务端从异步线程传递过来的数据,日志输出如下
--->name:say onClickage:1028Thread:Binder:5786_4