Android进程间通信(二):使用AIDL实现进程间通信
来源:互联网 发布:省市区县json数据 编辑:程序博客网 时间:2024/05/20 07:51
AIDL简单描述
AIDL全称是:Android Interface Define Language
因此传递对象, 只能把对象拆分成操作系统能理解的简单形式, 以达到跨界对象访问的目的. 在J2EE中,采用RMI的方式, 可以通过序列化传递对象. 在Android中, 则采用AIDL的方式. 理论上AIDL可以传递Bundle,实际上做起来却比较麻烦。
AIDL(AndRoid接口描述语言)是一种借口描述语言; 编译器可以通过aidl文件生成一段代码,通过预先定义的接口达到两个进程内部通信进程的目的。
如果需要在一个Activity中, 访问另一个Service中的某个对象, 需要先将对象转化成AIDL可识别的参数(可能是多个参数), 然后使用AIDL来传递这些参数, 在消息的接收端, 使用这些参数组装成自己需要的对象.AIDL的IPC的机制和COM或CORBA类似, 是基于接口的,但它是轻量级的。它使用代理类在客户端和实现层间传递值. 如果要使用AIDL, 需要完成2件事情:
1. 引入AIDL的相关类.; 2. 调用aidl产生的class.
AIDL实现进程间通信实战:
项目结构是:
在包名上点击右键新建.AIDL文件IMyAidlInterface.aidl,
package com.troy.aidldemo;// Declare any non-default types here with import statementsinterface IMyAidlInterface {//定义接口,两个方法; int add(int arg1,int arg2); int min(int arg1,int arg2);}
然后Make Project,会自动生成对应的IMyAidlInterface.java文件,这个后续再分析。
服务端CalcService.java核心代码如下:
public class CalcService extends Service{ private static final String TAG = "server"; public void onCreate(){ Log.e(TAG, "onCreate"); } public IBinder onBind(Intent t){ Log.e(TAG, "onBind"); return mBinder; } public boolean onUnbind(Intent intent){ Log.e(TAG, "onUnbind"); return super.onUnbind(intent); } public void onDestroy(){ Log.e(TAG, "onDestroy"); super.onDestroy(); } private final IMyAidlInterface.Stub mBinder = new IMyAidlInterface.Stub(){//IMyAidlInterface.Stub类是Binder的子类 @Override public int add(int x, int y) throws RemoteException{//实现接口中的加法 return x + y; } @Override public int min(int x, int y) throws RemoteException{//实现接口中的减法 return x - y; } };}
并且需要在清单文件中对CalcService进行注册服务。
<service android:name=".CalcService" android:process=".remote"> </service>
客户端代码MainActivity.java代码如下:
public class MainActivity extends AppCompatActivity { /** * 使用Binder实现通信(参见上一篇文章) */ private IBinder mPlusBinder; /** * 使用AIDL实现通信 */ private IMyAidlInterface mCalcAidl; private EditText et1,et2; private int arg1,arg2; private ServiceConnection mServiceConnPlus = new ServiceConnection(){ @Override public void onServiceConnected(ComponentName name, IBinder service){ Log.i("server", " onServiceConnected()"); //mPlusBinder = service; mCalcAidl = IMyAidlInterface.Stub.asInterface(service);// } @Override public void onServiceDisconnected(ComponentName name){ Log.i("server", "onServiceDisconnected()"); //mPlusBinder=null; mCalcAidl=null; } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et1=(EditText)findViewById(R.id.et1); et2=(EditText)findViewById(R.id.et2); } /**绑定服务*/ public void bindService(View view){ Intent intent = new Intent(this, CalcService.class); boolean plus=bindService(intent, mServiceConnPlus, Context.BIND_AUTO_CREATE); Toast.makeText(view.getContext(), "绑定服务"+plus, Toast.LENGTH_SHORT).show(); } /**取消链接*/ public void unbindService(View view){ try { unbindService(mServiceConnPlus); //mPlusBinder=null; Toast.makeText(view.getContext(), "解绑成功", Toast.LENGTH_SHORT).show(); } catch (Exception e) { e.printStackTrace(); Toast.makeText(view.getContext(), "未绑定", Toast.LENGTH_SHORT).show(); } } /**加法运算*/ public void add(View view) throws Exception{ if(!inspectData()){ return; } if (mCalcAidl != null){ int result = mCalcAidl.add(arg1, arg2); Toast.makeText(this, result + "", Toast.LENGTH_SHORT).show(); } else { Toast.makeText(this, "服务器被异常杀死,请重新绑定服务端", Toast.LENGTH_SHORT).show(); } } /**减法运算*/ public void subtraction(View view) throws Exception{ if(!inspectData()){ return; } if (mCalcAidl != null){ int result = mCalcAidl.min(arg1, arg2); Toast.makeText(this, result + "", Toast.LENGTH_SHORT).show(); } else{ Toast.makeText(this, "服务端未绑定或被异常杀死,请重新绑定服务端", Toast.LENGTH_SHORT).show(); } } /**检查数据*/ private boolean inspectData(){ //省略代码...}
最后运行的结果如图:
原理简单分析
在服务端,我们调用AIDL接口的代码是:
private final IMyAidlInterface.Stub mBinder = new IMyAidlInterface.Stub(){ @Override public int add(int x, int y) throws RemoteException{ return x + y; } @Override public int min(int x, int y) throws RemoteException{ return x - y; } };
再来系统看看我们写完AIDL文件之后,系统生成的java类是什么?我们找到generated文件夹下—>source—>aidl目录,如图所示:
找到IMyAidlInterface.java。先看看Stub静态类,Stub类IMyAidlInterface接口的静态抽象类,继承了Binder类,因此它是Binder的一个子类,同时实现了IMyAidlInterface接口。我们看看Stub类的定义:
public static abstract class Stub extends android.os.Binder implements com.troy.aidldemo.IMyAidlInterface
而IMyAidlInterface接口又是系统IInterface接口的子类:
public interface IMyAidlInterface extends android.os.IInterface
在客户端,我们实现了IMyAidlInterface接口的实例:
private IMyAidlInterface mCalcAidl;@Override public void onServiceConnected(ComponentName name, IBinder service){ Log.i("server", " onServiceConnected()"); //mPlusBinder = service; mCalcAidl = IMyAidlInterface.Stub.asInterface(service); }
其中调用了Stub抽象类的asInterface方法:
/** * Cast an IBinder object into an com.troy.aidldemo.IMyAidlInterface interface, * generating a proxy if needed. */ public static com.troy.aidldemo.IMyAidlInterface asInterface(android.os.IBinder obj){//该方法的作用是:将一个IBinder 对象转换成我们自定义的IMyAidlInterface 接口对象; if ((obj==null)) { return null; } android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR); if (((iin!=null)&&(iin instanceof com.troy.aidldemo.IMyAidlInterface))) { return ((com.troy.aidldemo.IMyAidlInterface)iin); } return new com.troy.aidldemo.IMyAidlInterface.Stub.Proxy(obj); }
同时在生成的java类中还会自动生成一个Proxy的静态代理类:
private static class Proxy implements com.troy.aidldemo.IMyAidlInterface{ private android.os.IBinder mRemote; Proxy(android.os.IBinder remote){ mRemote = remote; } @Override public android.os.IBinder asBinder(){ return mRemote; } public java.lang.String getInterfaceDescriptor(){ return DESCRIPTOR; } //中间的加法运算--会调用Stub的真正加法 @Override public int add(int arg1, int arg2) throws android.os.RemoteException{ //省略代码... } //中间的减法运算--会调用Stub的真正减法 @Override public int min(int arg1, int arg2) throws android.os.RemoteException { //省略代码... } static final int TRANSACTION_add = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); static final int TRANSACTION_min = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); }
我们来看看加法的运算流程到底是如何:
1 第一步
在客户端的EditText获取用户输入的参数,作非空判断:
/**检查数据*/ private EditText et1,et2; private int arg1,arg2; private boolean inspectData(){ if(TextUtils.isEmpty(et1.getText().toString())||TextUtils.isEmpty(et2.getText().toString())){ Toast.makeText(this, "请填全两个参数!", Toast.LENGTH_SHORT).show(); return false; } arg1=Integer.parseInt(et1.getText().toString()); arg2=Integer.parseInt(et2.getText().toString()); return true; }
2 第二步
在客户端调用IMyAidlInterface的add方法:
int result = mCalcAidl.add(arg1, arg2);
3 第三步
会执行代理类Proxy的add方法:
private static class Proxy implements com.troy.aidldemo.IMyAidlInterface{ @Override public int add(int arg1, int arg2) throws android.os.RemoteException{ android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); int _result; try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeInt(arg1); _data.writeInt(arg2); //mRemote就是一个IBinder对象 mRemote.transact(Stub.TRANSACTION_add, _data, _reply, 0); _reply.readException(); _result = _reply.readInt(); } finally { _reply.recycle(); _data.recycle(); } return _result; }
4 第四步
会调用Stub的add方法,代码如下:
@Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException{ switch (code){ case INTERFACE_TRANSACTION: { reply.writeString(DESCRIPTOR); return true; } case TRANSACTION_add: { data.enforceInterface(DESCRIPTOR); int _arg0; _arg0 = data.readInt(); int _arg1; _arg1 = data.readInt(); int _result = this.add(_arg0, _arg1); reply.writeNoException(); reply.writeInt(_result); return true; } }
5 第五步
在服务端会回调mBinder 的add方法,这是真正实现加法运算的地方,:
private final IMyAidlInterface.Stub mBinder = new IMyAidlInterface.Stub(){ @Override public int add(int x, int y) throws RemoteException{ return x + y; } };
同理,减法运算的调用流程也是一样:
客户端参数-->Proxy代理类--->Stub类--->CalcService中的mBinder
AIDL中实现进程间通信的流程图分析为:
以上,就完成了使用AIDL来实现进程间的通信。
总结:
通过本DEMO需要掌握以下几点:
(1)如何写IMyAidlInterface.aidl文件,并能分析系统生成的IMyAidlInterface.java代码,理解其中的onTransact()方法;
(2)在服务端要实现IMyAidlInterface.Stub实例及接口方法,并理解Stub类与Binder类的关系。
(3)客户端与服务端的连接,bindService(intent, mServiceConnPlus, Context.BIND_AUTO_CREATE);
(4)在客户端实现IMyAidlInterface接口对象,并理解IMyAidlInterface.Stub.asInterface(service);
学习资料致谢:
1 Android aidl Binder框架浅析
http://blog.csdn.net/lmj623565791/article/details/38461079/
2 Binder牌胶水
http://www.jianshu.com/p/3d053abba04b
3 Binder学习指南
http://www.jianshu.com/p/af2993526daf
4 Android Bander设计与实现 - 设计篇
http://blog.csdn.net/universus/article/details/6211589
5 Android进程间通信(IPC)机制Binder简要介绍和学习计划
http://blog.csdn.net/luoshengyang/article/details/6618363
6 【Android】由浅到深理解AIDL
http://blog.csdn.net/realizelizijun2013/article/details/50989878
7 android跨进程通信(IPC):使用AIDL
http://blog.csdn.net/singwhatiwanna/article/details/17041691?reload
- Android进程间通信(二):使用AIDL实现进程间通信
- android 使用AIDL实现进程间通信
- Android使用AIDL实现进程间通信
- Android使用AIDL实现进程间通信
- Android使用AIDL实现进程间通信
- Android使用AIDL实现进程间通信
- Android使用AIDL实现进程间通信
- Android使用AIDL实现进程间通信
- Android进程间通信(二):AIDL
- Android 进程间通信AIDL讲解 二
- Android Aidl实现进程间通信
- android笔记 AIDL 实现进程间通信
- Android AIDL IPC实现进程间通信
- AIDL实现Android进程间通信实例
- android AIDL实现进程间的通信
- AIDL实现Android进程间的通信
- 使用AIDL实现进程间的通信
- 使用AIDL实现进程间的通信
- 解决eclipse中Adb connection Error:远程主机强迫关闭了一个现有的连接 的问题
- 第128课: Spark Streaming源码经典解读系列之三:JobScheduler工作内幕源
- Android热修复学习(二)
- 求二叉树的宽度C语言版
- ovs-dpdk bridge internal port 访问延迟问题, fix
- Android进程间通信(二):使用AIDL实现进程间通信
- 差分约束系统
- Linux 挂载 新磁盘
- Maven的安装和settings.xml的配置
- unity中建立 Socket 简单通信
- java与C++的区别
- 消除数组相同项,把该项之后的前移
- HITS算法思想
- Avoid object allocations during draw/layout operations (preallocate and reuse instead)