【Android基础知识】AIDL跨进程调用
2016-07-13 16:42
447 查看
AIDL(Android interface definition language)Android接口定义语言。
Android 系统中,各应用程序都运行在自己的进程中,进程之间一般无法直接进行数据交换,为了实现这种跨进程通信,Android提供了AIDL Service.
客户端访问Service时,Android并不是直接返回Service对象给客户端,只是将Service的代理(IBinder对象)通过onBind方法返回给客户端。因此Android的AIDL的远程接口的实现类就是那个IBinder实现类。
和绑定本地Service不同的是,本地会把IBinder对象本身返回给onServiceConnected方法的第二个参数。但是远程Service只是将IBinder对象的代理传给客户端ServiceConnected的第二个参数。当客户端获取了远程Service的IBinder对象的代理之后,接下来就可以通过IBinder对象去回调远程Service的属性或方法了。
AIDL使用实例:实现服务器端启动一个服务,随机更改颜色和体重,客户端获取这些信息
2.android会自动生成一个类,类里有stub静态内部类实现了IBinder和Aidl接口。
3.创建我们自己的Service,实现stub类,返回IBinder对象。
4.创建客户端程序,创建aidl接口文件,注意文件名和包名要完全相同,同样自动生成一个类。
5.绑定service,获取代理。
AIDL默认支持的数据类型
基本数据类型(除short)
String、CharSequence、List、Map、Parcelable(序列化)
Android5.0 以后Service不能隐式启动,service Intent must be explitict
源码中是这样的 (源码位置:sdk/sources/android-21/android/app/ContextImpl.java)
有两种解决方式:
1.设置package和packagename
1.首先Servier去实现了Stub(Binder)内部类,实现了里面的方法并返回一个IBinder对象。
2.客户端调用Stub.asInterface(Binder),这里其实返回的是代理类Proxy对象 return new com.aidl.ICat.Stub.Proxy(obj),该方法代码如下:
是Stub类对象,我们的Service中 继承了Stub类并且实现了里面的方法,这里方法会被回调this.getColor(),把结果发送给客户端。
Android 系统中,各应用程序都运行在自己的进程中,进程之间一般无法直接进行数据交换,为了实现这种跨进程通信,Android提供了AIDL Service.
客户端访问Service时,Android并不是直接返回Service对象给客户端,只是将Service的代理(IBinder对象)通过onBind方法返回给客户端。因此Android的AIDL的远程接口的实现类就是那个IBinder实现类。
和绑定本地Service不同的是,本地会把IBinder对象本身返回给onServiceConnected方法的第二个参数。但是远程Service只是将IBinder对象的代理传给客户端ServiceConnected的第二个参数。当客户端获取了远程Service的IBinder对象的代理之后,接下来就可以通过IBinder对象去回调远程Service的属性或方法了。
AIDL使用实例:实现服务器端启动一个服务,随机更改颜色和体重,客户端获取这些信息
服务器端
1.建立aidl文件package com.aidl; interface ICat { String getColor(); double getWeight(); }
2.android会自动生成一个类,类里有stub静态内部类实现了IBinder和Aidl接口。
/* * This file is auto-generated. DO NOT MODIFY. * Original file: E:\\AIDLService\\src\\com\\aidl\\ICat.aidl */ package com.aidl; public interface ICat extends android.os.IInterface { /** Local-side IPC implementation stub class. */ public static abstract class Stub extends android.os.Binder implements com.aidl.ICat { private static final java.lang.String DESCRIPTOR = "com.aidl.ICat"; /** Construct the stub at attach it to the interface. */ public Stub() { this.attachInterface(this, DESCRIPTOR); } /** * Cast an IBinder object into an com.aidl.ICat interface, * generating a proxy if needed. */ public static com.aidl.ICat asInterface(android.os.IBinder obj) { if ((obj==null)) { return null; } android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR); if (((iin!=null)&&(iin instanceof com.aidl.ICat))) { return ((com.aidl.ICat)iin); } return new com.aidl.ICat.Stub.Proxy(obj); } @Override public android.os.IBinder asBinder() { return this; } @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_getColor: { data.enforceInterface(DESCRIPTOR); java.lang.String _result = this.getColor(); reply.writeNoException(); reply.writeString(_result); return true; } case TRANSACTION_getWeight: { data.enforceInterface(DESCRIPTOR); double _result = this.getWeight(); reply.writeNoException(); reply.writeDouble(_result); return true; } } return super.onTransact(code, data, reply, flags); } private static class Proxy implements com.aidl.ICat { 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; } @Override public java.lang.String getColor() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); java.lang.String _result; try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_getColor, _data, _reply, 0); _reply.readException(); _result = _reply.readString(); } finally { _reply.recycle(); _data.recycle(); } return _result; } @Override public double getWeight() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); double _result; try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_getWeight, _data, _reply, 0); _reply.readException(); _result = _reply.readDouble(); } finally { _reply.recycle(); _data.recycle(); } return _result; } } static final int TRANSACTION_getColor = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); static final int TRANSACTION_getWeight = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); } public java.lang.String getColor() throws android.os.RemoteException; public double getWeight() throws android.os.RemoteException; }
3.创建我们自己的Service,实现stub类,返回IBinder对象。
/* * 定义到AIDL接口之后,需要定义一个Service实现类,该Service的onBind方法 * 所返回的IBinder对象时ADT所生成的ICat.stub的子类的实例。 */ public class AidlService extends Service{ private CatBinder catBinder; Timer timer = new Timer(); String[] colors = new String[]{ "红色", "黄色", "黑色" }; double [] weights = new double[]{ 2.3, 3.1, 1.58 }; private String color; private double weight; //继承Stub,也就是实现了ICat接口,并实现了IBinder接口 public class CatBinder extends Stub{ @Override public String getColor() throws RemoteException { // TODO Auto-generated method stub return color; } @Override public double getWeight() throws RemoteException { // TODO Auto-generated method stub return weight; } } @Override public void onCreate() { // TODO Auto-generated method stub super.onCreate(); catBinder = new CatBinder(); timer.schedule(new TimerTask() { @Override public void run() { // TODO Auto-generated method stub //随机的改变Service组件内color、weight的属性的值 int rand = (int)(Math.random()*3); color = colors[rand]; weight = weights[rand]; } }, 0,800); } @Override public void onDestroy() { // TODO Auto-generated method stub timer.cancel(); } @Override public IBinder onBind(Intent intent) { // TODO Auto-generated method stub /* * 返回catBinder对象,绑定本地,直接传递对象 * 绑定远程,传递代理。 */ return catBinder; } }
4.创建客户端程序,创建aidl接口文件,注意文件名和包名要完全相同,同样自动生成一个类。
5.绑定service,获取代理。
public class MainActivity extends Activity { private ICat catService; private Button get; EditText color,weight; private ServiceConnection connection = new ServiceConnection() { @Override public void onServiceDisconnected(ComponentName name) { // TODO Auto-generated method stub catService = null; } @Override public void onServiceConnected(ComponentName name, IBinder service) { // TODO Auto-generated method stub //获取远程Service的onBind方法返回的对象的代理 catService = ICat.Stub.asInterface(service); } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(save db37 dInstanceState); setContentView(R.layout.activity_main); get = (Button)findViewById(R.id.button1); color = (EditText)findViewById(R.id.editText1); weight = (EditText)findViewById(R.id.editText2); //创建所需绑定的Service的intent Intent intent = new Intent(); intent.setAction("com.aidl.AIDL_Service"); Intent eintent = new Intent(getExplicitIntent(getBaseContext(),intent)); //绑定远程Service bindService(eintent, connection, Service.BIND_AUTO_CREATE); get.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub try{ color.setText(catService.getColor()); weight.setText(catService.getWeight()+""); }catch(RemoteException e){ e.printStackTrace(); } } }); } //将隐式intent装换为显示intent public static Intent getExplicitIntent(Context context, Intent implicitIntent) { // Retrieve all services that can match the given intent PackageManager pm = context.getPackageManager(); List<ResolveInfo> resolveInfo = pm.queryIntentServices(implicitIntent, 0); // Make sure only one match was found if (resolveInfo == null || resolveInfo.size() != 1) { return null; } // Get component info and create ComponentName ResolveInfo serviceInfo = resolveInfo.get(0); String packageName = serviceInfo.serviceInfo.packageName; String className = serviceInfo.serviceInfo.name; ComponentName component = new ComponentName(packageName, className); // Create a new intent. Use the old one for extras and such reuse Intent explicitIntent = new Intent(implicitIntent); // Set the component to be explicit explicitIntent.setComponent(component); return explicitIntent; } @Override protected void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); this.unbindService(connection); } }
AIDL默认支持的数据类型
基本数据类型(除short)
String、CharSequence、List、Map、Parcelable(序列化)
Android5.0 以后Service不能隐式启动,service Intent must be explitict
源码中是这样的 (源码位置:sdk/sources/android-21/android/app/ContextImpl.java)
private void validateServiceIntent(Intent service) { if (service.getComponent() == null && service.getPackage() == null) { if (getApplicationInfo().targetSdkVersion >= Build.VERSION_CODES.LOLLIPOP) { IllegalArgumentException ex = new IllegalArgumentException( "Service Intent must be explicit: " + service); throw ex; } else { Log.w(TAG, "Implicit intents with startService are not safe: " + service + " " + Debug.getCallers(2, 3)); } } }
有两种解决方式:
1.设置package和packagename
Intent mIntent = new Intent(); mIntent.setAction("XXX.XXX.XXX");//你定义的service的action mIntent.setPackage(getPackageName());//这里你需要设置你应用的包名 context.startService(mIntent);2.将隐式转换为显式启动
public static Intent getExplicitIntent(Context context, Intent implicitIntent) { // Retrieve all services that can match the given intent PackageManager pm = context.getPackageManager(); List<ResolveInfo> resolveInfo = pm.queryIntentServices(implicitIntent, 0); // Make sure only one match was found if (resolveInfo == null || resolveInfo.size() != 1) { return null; } // Get component info and create ComponentName ResolveInfo serviceInfo = resolveInfo.get(0); String packageName = serviceInfo.serviceInfo.packageName; String className = serviceInfo.serviceInfo.name; ComponentName component = new ComponentName(packageName, className); // Create a new intent. Use the old one for extras and such reuse Intent explicitIntent = new Intent(implicitIntent); // Set the component to be explicit explicitIntent.setComponent(component); return explicitIntent; }Aidl的底层实现原理
1.首先Servier去实现了Stub(Binder)内部类,实现了里面的方法并返回一个IBinder对象。
2.客户端调用Stub.asInterface(Binder),这里其实返回的是代理类Proxy对象 return new com.aidl.ICat.Stub.Proxy(obj),该方法代码如下:
public static com.aidl.ICat asInterface(android.os.IBinder obj) { if ((obj==null)) { return null; } android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR); if (((iin!=null)&&(iin instanceof com.aidl.ICat))) { return ((com.aidl.ICat)iin); } return new com.aidl.ICat.Stub.Proxy(obj); }Proxy(IBinder代理类代码如下)里面实现了我们接口中定义的两个方法,getColor和getWeight
private static class Proxy implements com.aidl.ICat { 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; } @Override public java.lang.String getColor() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); java.lang.String _result; try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_getColor, _data, _reply, 0); _reply.readException(); _result = _reply.readString(); } finally { _reply.recycle(); _data.recycle(); } return _result; } @Override public double getWeight() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); double _result; try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_getWeight, _data, _reply, 0); _reply.readException(); _result = _reply.readDouble(); } finally { _reply.recycle(); _data.recycle(); } return _result; } } static final int TRANSACTION_getColor = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); static final int TRANSACTION_getWeight = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); }3.客户端利用代理对象Proxy 调用了getColor()方法,Proxy类实现的getColor()方法,该方法会把我们调用的方法的ID Stub.TRANSACTION_getColor 传递给底层IPC
mRemote.transact(Stub.TRANSACTION_getColor, _data, _reply, 0);4.到达服务器端调用onTransact方法,该方法会根据刚才传入的方法ID通过switch case 去匹配到TRANSACTION_getColor ,然后会调用 this.getColor()方法,这里的this就
是Stub类对象,我们的Service中 继承了Stub类并且实现了里面的方法,这里方法会被回调this.getColor(),把结果发送给客户端。
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_getColor: { data.enforceInterface(DESCRIPTOR); java.lang.String _result = this.getColor(); reply.writeNoException(); reply.writeString(_result); return true; } case TRANSACTION_getWeight: { data.enforceInterface(DESCRIPTOR); double _result = this.getWeight(); reply.writeNoException(); reply.writeDouble(_result); return true; } } return super.onTransact(code, data, reply, flags); }
相关文章推荐
- 使用C++实现JNI接口需要注意的事项
- Android IPC进程间通讯机制
- Android Manifest 用法
- [转载]Activity中ConfigChanges属性的用法
- Android之获取手机上的图片和视频缩略图thumbnails
- Android之使用Http协议实现文件上传功能
- Android学习笔记(二九):嵌入浏览器
- android string.xml文件中的整型和string型代替
- i-jetty环境搭配与编译
- android之定时器AlarmManager
- android wifi 无线调试
- Android Native 绘图方法
- Android java 与 javascript互访(相互调用)的方法例子
- android 代码实现控件之间的间距
- android FragmentPagerAdapter的“标准”配置
- Android"解决"onTouch和onClick的冲突问题
- android:installLocation简析
- android searchView的关闭事件
- SourceProvider.getJniDirectories