上一篇我们在学习bindService时发现了一个问题,即我们在客户端通过bindservice()方法绑定服务之后,服务器给我们返回了一个IBinder对象,同时我们还在服务器里面写了一个继承了Binder类的内部类等等。我们做了那么多的操作,发现其实只是在本地进程中调用了一下本地方法。逗我玩呢?
事实上,那仅仅只是Binder在本地进程中的一种简单用法,关于强大的Binder机制,图文详解 Android Binder跨进程通信的原理,那么今天,我们就讲讲Binder的跨进程通信。
说起跨进程之前,那么我们首先要理解一个概念:什么是多进程?
在android系统中,我们知道,打开一个app,意味着我们就打开了一个进程(普遍来说),然后在这个进程中又会有许多线程。即一个app就是一个进程。当然了,android系统为了保证应用的流畅性给我们提供了一个android:process的属性,这可以让我们在一个app中跑多个进程,分摊压力,保证应用的流畅性和可用性。当我们给组件元素:、、、设置一个android:process属性时,这个组件就跑在了自己定义的进程中了。这就是所谓的多进程。
开启多进程
在组件配置中设置 android:process 属性,即可让组件跑在自己的进程中
关于多进程有两种设置方法 android:process=":xxxx" 和 android:process="com.demo.xxxx" 两种,以 : 开头的命名进程为私有进程其它应用不能通过ShareUID方式和它跑在一个进程中,以com.demo.xxxx命名的进程其它应用可以通过ShareUID方式和它跑在一个进程中。
使用Binder进行进程间通信
一般来讲,我们有三种方式可以获得IBinder的对象:继承Binder类,使用Messenger类,使用AIDL。今天我们主要讲一下AIDL,因为Messenger是AIDL的封装,AIDL会了。Messenger也就会使用。
关于使用AIDL来进行进程间的通信步骤:
1. 在项目 src 目录下添加aidl的文件夹
2. 创建自己的aidl文件以及实体类的映射aidl文件
3. 重新构建工程,会在 build/generated/source/aidl/你的 flavor/ 生成一个关于你的aidl文件的java文件。这个文件就是跨进程 Client 和 Server 的通信媒介。
4. 服务端的代码编写
public class TestAidlServiceextends Service {
private static final StringTAG ="TestAidlService";
private ArrayListmBooks;
private IBindermIBinder =new IMyAidlInterface.Stub() {
@Override
public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,double aDouble, String aString)throws RemoteException {
}
@Override
public void addBook(Book book)throws RemoteException {
mBooks.add(book);
}
@Override
public ListgetBookList()throws RemoteException {
return mBooks;
}
};
/**
* 客户端与服务端绑定时的回调,返回 mIBinder 后客户端就可以通过它远程调用服务端的方法,即实现了通讯
* @param intent
* @return
*/
@Nullable
@Override
public IBinderonBind(Intent intent) {
mBooks =new ArrayList<>();
return mIBinder;
}
}
5.客户端代码
public class TestActivityextends BaseActivity {
private IMyAidlInterfacemIMyAidlInterface;
private boolean mIsBinder =false;
@BindView(R.id.button)
ButtonmButton;
@BindView(R.id.tv_result)
TextViewmText;
@Override
protected int getRootView() {
return R.layout.activity_con;
}
@OnClick({R.id.button})
public void onBindClick(View view){
switch (view.getId()){
case R.id.button:
try {
mIMyAidlInterface.addBook(new Book("sss"));
List bookList =mIMyAidlInterface.getBookList();
mText.setText(bookList.toString());
}catch (RemoteException e) {
e.printStackTrace();
}
break;
}
}
@Override
protected void initData() {
Intent intent =new Intent(getApplicationContext(),TestAidlService.class);
//客户段绑定服务
bindService(intent,TestAidlServiceConnection, Context.BIND_AUTO_CREATE);
}
private ServiceConnectionTestAidlServiceConnection =new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
mIMyAidlInterface = IMyAidlInterface.Stub.asInterface(service);
mIsBinder =true;
}
@Override
public void onServiceDisconnected(ComponentName name) {
mIMyAidlInterface =null;
mIsBinder =false;
}
};
}
6. 在AndroidManifest.xml文件中配置我们的服务
android:enabled="true" android:exported="true" android:process=":aidl"/> 当我们在手机上运行我们的代码时,我们可以很清晰的看到我们的代码是跑在两个进程中的 最后放上我们的演示结果 绑定服务时需要传入全局的上下文,因为是访问另一个进程。 Intent intent =new Intent(getApplicationContext(),TestAidlService.class); 对象序列化时需继承parcelable,不然在build时com.syz.base.lib.aidl.Book.CREATOR.createFromParcel(data)这个地方会报错。 结果
注意点: