关于IPC应该不用多介绍了,AndroID系统中的进程之间不能共享内存,那么如果两个不同的应用程序之间需要通讯怎么办呢?比如公司的一个项目要更新,产品的需求是依附于当前项目开发一个插件,但是呢这个插件功能以及界面比较复杂,不能和当前项目在一个进程中,同时呢,还要用到当前项目中已经写好了的一些东西,那么因为新开发的依附于当前项目的插件和当前项目不是一个进程,因此不能共享内存,就出现了问题,于是,需要提供一些机制在不同进程之间进行数据通信,这个机制就是AIDL了。
一、一个androID中AIDL的简单例子
假如是这样,现在有一个项目中提供了比较成熟的计算的方法,而现在我想开发一款软件其中一个模块想用到一个计算类,而我又不想重新写了,那么就可以通过AIDL实现啦。假设,已经开发完成的那个已经提供了比较成熟的计算类的程序叫AIDLCalculateDemoServer(相当于服务器),而我要写的程序叫AIDLCalculateDemoClIEnt(相当于客户端),类似与客户端服务器模式。首先至关的看下工程结构图:
图1-1 服务器 图1-2 客户端
现在假设自己写的程序要调用服务端的运算界面,输入num1和num2,进行远程运算,调用服务端的接口,服务端运算好之后,返回结果给客户端,效果图如下:
然后来看看实现,首先需要定义AIDL接口,客户端和服务器端都要定义,并且要在同一包中,也就是图1-1和图1-2 com.example.aIDl.calculate中的CalculateInterface,其中的代码如下:
package com.example.aIDl.calculate;interface CalculateInterface { double doCalculate(double a,double b); }
编译发现,目录结构如图1-1和图1-2中gen/com.example.aIDl.calculate中多了CalculateInterface.java文件,内容如下:
package com.example.aIDl.calculate; interface CalculateInterface { double doCalculate(double a,double b); }
定义好接口就是要看服务端和客户端的代码啦,其中服务端主要看CalculateService代码,这个一个继承Service的类,在其中对AIDL中的接口进行赋予实际意义,如下:
package com.example.calculate;import com.example.aIDl.calculate.CalculateInterface;import com.example.aIDl.calculate.CalculateInterface.Stub;import androID.app.Service;import androID.content.Intent;import androID.os.IBinder;import androID.os.remoteexception;import androID.util.Log;public class CalculateService extends Service { private static final String TAG = "CalculateService"; @OverrIDe public IBinder onBind(Intent arg0) { // Todo auto-generated method stub logE("onBind()"); return mBinder; } @OverrIDe public voID onCreate() { // Todo auto-generated method stub logE("onCreate()"); super.onCreate(); } @OverrIDe public voID onStart(Intent intent,int startID) { // Todo auto-generated method stub logE("onStart()"); super.onStart(intent,startID); } @OverrIDe public boolean onUnbind(Intent intent) { // Todo auto-generated method stub logE("onUnbind()"); return super.onUnbind(intent); } @OverrIDe public voID onDestroy() { // Todo auto-generated method stub logE("onDestroy()"); super.onDestroy(); } private static voID logE(String str) { Log.e(TAG,"--------" + str + "--------"); } private final CalculateInterface.Stub mBinder = new CalculateInterface.Stub() { @OverrIDe public double doCalculate(double a,double b) throws remoteexception { // Todo auto-generated method stub Log.e("Calculate","远程计算中"); Calculate calculate = new Calculate(); double answer = calculate.calculateSum(a,b); return answer; } };}
然后可以看看,关键的服务都提供完毕,那么在客户端是怎么访问的呢,要进行绑定服务和一个ServiceConnection类完成,如下:
package com.example.calculate;import androID.app.Activity;import androID.content.Componentname;import androID.content.Context;import androID.content.Intent;import androID.content.ServiceConnection;import androID.graphics.color;import androID.os.Bundle;import androID.os.IBinder;import androID.os.remoteexception;import androID.util.Log;import androID.vIEw.VIEw;import androID.Widget.button;import androID.Widget.EditText;import androID.Widget.TextVIEw;import com.example.aIDl.calculate.CalculateInterface;import com.example.aIDlcalculatedemoclIEnt.R;public class CalculateClIEnt extends Activity { private static final String TAG = "CalculateClIEnt"; private button btncalculate; private EditText etNum1; private EditText etNum2; private TextVIEw tvResult; private CalculateInterface mService; private ServiceConnection mServiceConnection = new ServiceConnection() { @OverrIDe public voID onServicedisconnected(Componentname name) { // Todo auto-generated method stub logE("disconnect service"); mService = null; } @OverrIDe public voID onServiceConnected(Componentname name,IBinder service) { // Todo auto-generated method stub logE("connect service"); mService = CalculateInterface.Stub.asInterface(service); } }; @OverrIDe protected voID onCreate(Bundle savedInstanceState) { // Todo auto-generated method stub super.onCreate(savedInstanceState); setContentVIEw(R.layout.main); Bundle args = new Bundle(); Intent intent = new Intent("com.example.calculate.CalculateService"); intent.putExtras(args); bindService(intent,mServiceConnection,Context.BIND_auto_CREATE); etNum1 = (EditText) findVIEwByID(R.ID.et_num_one); etNum2 = (EditText) findVIEwByID(R.ID.et_num_two); tvResult = (TextVIEw) findVIEwByID(R.ID.tv_result); btncalculate = (button) findVIEwByID(R.ID.btn_cal); btncalculate.setonClickListener(new VIEw.OnClickListener() { @OverrIDe public voID onClick(VIEw v) { // Todo auto-generated method stub logE("开始远程运算"); try { double num1 = Double.parseDouble(etNum1.getText().toString()); double num2 = Double.parseDouble(etNum2.getText().toString()); String answer = "计算结果:" + mService.doCalculate(num1,num2); tvResult.setTextcolor(color.BLUE); tvResult.setText(answer); } catch (remoteexception e) { } } }); } private voID logE(String str) { Log.e(TAG,"--------" + str + "--------"); }}
如此一来,大功已经基本告成,最后,我们在来看看服务端的配置文件吧:
<?xml version="1.0" enCoding="utf-8"?><manifest xmlns:androID="http://schemas.androID.com/apk/res/androID" package="com.example.aIDlcaculatedemoserver" androID:versionCode="1" androID:versionname="1.0" > <uses-sdk androID:minSdkVersion="8" androID:targetSdkVersion="17" /> <application androID:allowBackup="true" androID:icon="@drawable/ic_launcher" androID:label="@string/app_name" androID:theme="@style/Apptheme" > <activity androID:name="com.example.aIDlcaculatedemoserver.MainActivity" androID:label="@string/app_name" > <intent-filter> <action androID:name="androID.intent.action.MAIN" /> <category androID:name="androID.intent.category.LAUNCHER" /> </intent-filter> </activity> <service androID:name="com.example.calculate.CalculateService"> <intent-filter> <action androID:name="com.example.calculate.CalculateService" /> </intent-filter> </service> </application></manifest>
二、写AIDL注意事项
1. 客户端和服务端的AIDL接口文件所在的包必须相同
2. 需要一个Service类的配合
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。
总结以上是内存溢出为你收集整理的android中的AIDL进程间通信示例全部内容,希望文章能够帮你解决android中的AIDL进程间通信示例所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)