最近在看Binder的实现机制,但是感觉总不是很清晰,所以想理解他的实现原理,先暂时学会怎么使用。Binder是android中的一种进程间通信机制,
android的底层是Linux系统,Linux系统中进程间通信方式主要有以下几种:
1、socket,即客户端服务器模式。所以我们客户端程序通过socket和服务器端程序进行通信,其实就是进程间通信的一个实例。
2、传统的消息队列机制
3、共享内存机制
4、信号量机制
在android系统中,由于安全性等一些其他因素影响,所以android的开发者们实现了另外一种进程间通信方式Binder,android中所有的进程间通信方式
都是通过Binder以及以上4种方式来实现的,ContentProvider实现的进程间数据共享方式就是通过共享内存以及Binder方式来实现进程间通信的,具体
的详细解释可以参考老罗的这篇博客:http://blog.csdn.net/luoshengyang/article/details/6946067。android的RPC机制(远程过程调用)也是通过Binder
机制来实现的,RPC其实就是IPC机制的一种实现方式,IPC即Interprocess communication 进程间通信,实现IPC有很多方式,但是有时候我们并不满
足于仅仅实现数据通信,有时候我们还需要去调用其它进程间的方法或对象,这时就需要RPC机制了,而android中就是通过AIDL来实现这样一种RPC
机制。具体关于RPC的定义可以google去看wiki的介绍。下面我们就来看看android的RPC机制是怎样通过AIDL(接口定义语言)来实现进程间的方法或对
象调用的。
1、新建一个AIDL文件RemoteServiceAidl.aidl,定义几个方法,分别以基本类型数据和对象作为参数。定义好之后会在工程的gen目录下生成相同的包
名以及相同文件名的java文件RemoteServiceAidl.java,他的代码我就不贴出来了。
package com.lonuery.aidl; import com.lonuery.remote.GroupInfo; interface RemoteServiceAidl { int calculate(int num1,int num2); void checkInfo(String str,double data); String save(in GroupInfo groupInfo); }
2、由于AIDL中传递基本数据类型和对象的方式不同,传递对象就必须实现Parcelable接口,接下来定义GroupInfo类
public class GroupInfo implements Parcelable{ String name,groupId,memCount; public GroupInfo(String name,String groupId,String memCount){ this.name = name; this.groupId = groupId; this.memCount = memCount; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getGroupId() { return groupId; } public void setGroupId(String groupId) { this.groupId = groupId; } public String getMemCount() { return memCount; } public void setMemCount(String memCount) { this.memCount = memCount; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel parcel, int arg1) { //参数的写入是什么顺序,那么下面参数的读取就应该是什么顺序。参数的写入顺序不对,会导致读取的参数顺序不对 parcel.writeString(this.name); parcel.writeString(this.groupId); parcel.writeString(this.memCount); } //如果要传递对象那么这个类中就必须创建一个名字为CREATOR的Creator对象。 public static final Parcelable.Creator<GroupInfo> CREATOR = new Parcelable.Creator<GroupInfo>() { @Override public GroupInfo createFromParcel(Parcel parcel) { return new GroupInfo(parcel.readString(), parcel.readString(), parcel.readString()); } @Override public GroupInfo[] newArray(int size) { return new GroupInfo[size]; } }; }
3、在GroupInfo.java相同包下定义GroupInfo.aidl文件。
package com.lonuery.remote; parcelable GroupInfo;
3、实现RemoteServiceAidl.Stub类,并且定义一个Service以供其他程序来远程绑定,返回Stub的对象。
public class RemoteService extends Service{ String TAG = "RemoteService"; AidlImpl aidlService = new AidlImpl(); @Override public IBinder onBind(Intent arg0) { Log.v(TAG, "onBind"); return aidlService; } class AidlImpl extends RemoteServiceAidl.Stub{ @Override public int calculate(int num1, int num2) throws RemoteException { int result = num1*num2; return result; } @Override public void checkInfo(String str, double data) throws RemoteException { if(str!=null){ Log.v("checkInfo", "输出"); } } @Override public String save(GroupInfo groupInfo) throws RemoteException { String str = null; if(groupInfo!=null){ if(groupInfo.getGroupId().equals("电话")){ str = "地方金额1"; }else if(groupInfo.getMemCount().equals("电话")){ str = "地方金额2"; }else if(groupInfo.getName().equals("电话")){ str = "地方金额3"; } } return str; } } }
1、在新建的调用app中新建和RemoteServiceAidl.aidl相同的包名,并将RemoteServiceAidl.aidl复制到这个包中。同样,在gen目录下也会自动生成
RemoteServiceAidl.java的文件。
2、新建和GroupInfo.java相同的包名并将GroupInfo.java和GroupInfo.aidl文件复制到此包中。
3、绑定远程服务,利用远程服务对象,调用远程方法。
public class MainActivity extends Activity implements OnClickListener{ Button btn1; RemoteServiceAidl remoteService; TextView tv; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); btn1 = (Button)findViewById(R.id.button1); btn1.setOnClickListener(this); tv = (TextView)findViewById(R.id.textView1); bindService(new Intent(RemoteServiceAidl.class.getName()), connection, Context.BIND_AUTO_CREATE); } @Override public void onClick(View view) { if(view.getId()==R.id.button1){ try { int result=0; GroupInfo info = new GroupInfo("", "电话", ""); String str=null; try { result = remoteService.calculate(6, 8); str = remoteService.save(info); } catch (DeadObjectException e) { e.printStackTrace(); } tv.setText("result:"+result); String text = tv.getText().toString() + "Id:"+str; tv.setText(text); } catch (RemoteException e) { e.printStackTrace(); } } } @Override protected void onDestroy() { unbindService(connection); super.onDestroy(); } private ServiceConnection connection = new ServiceConnection() { @Override public void onServiceDisconnected(ComponentName arg0) { remoteService = null; } @Override public void onServiceConnected(ComponentName arg0, IBinder binder) { remoteService = RemoteServiceAidl.Stub.asInterface(binder); } }; }
4、在AndroidManifest.xml中对所创建的服务进行注册。
接下来我们分别安装远程app,和调用app,来看一下效果。
好了android中aidl的使用实例就完成了,接下来我们总结一下:
1、RemoteService只是一个代理的角色,他的作用就是让其他的进程来绑定他,然后通过他进行远程方法调用,真正的调用其实是在
RemoteServiceAidl.Stub的实现类AidlImpl中实现的。
2、将AidlImpl类的对象返回给RemoteService的绑定接口onBind,远程调用就是通过这个接口来获取AidlImpl对象,继而实现方法的调用。
3、远程进程通过bindService方法绑定远程服务RemoteService。
实例已经上传到csdn上http://download.csdn.net/detail/zkw12358/7362967