|
这两天在看面试题,上面是这样问的:Android Service和Binder、AIDL你都熟练吗 ?
说实话,Service知道,但是没怎么用过,好像最典型的例子就是用在音乐播放器上了,Binder和那个AIDL是什么东西呢?为了弄懂它,我今天开始学习AIDL。
在网上找了几个例子,运行了一下,看了下源码,还不是很懂,继续看看。其实, 很简单,众所周知,在当前Activity中启动service除了,1. startService(),2.bindService() 外, 还有第三种启动方式,RPC远程进程访问方式启动Service。其实AIDL不是为了有更多的方法去启动服务,而是提供了一种跨进程通信的机制。这个大家一定要清楚,就像好多朋友问,AIDL有什么用,我学它干嘛?我只能这样回答:小的应用没必要用它,除非你是一个愿意把代码写到及其艺术感的人,或者你闲的没事干,或者你就想研究它。那么,学它干嘛,我想是这样的,了解这样一种机制,在需要用到的时候用它可以解决很大的问题,回答完毕。大家请往下看。
详情请参看:http://buaadallas.blog.51cto.com/399160/372090
在Android中, 每个应用程序都可以有自己的进程. 在写UI应用的时候, 经常要用到Service. 在不同的进程中, 怎样传递对象呢? 显然, java中不允许跨进程内存共享. 因此传递对象, 只能把对象拆分成操作系统能理解的简单形式, 以达到跨界对象访问的目的. 在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语法很简单,可以用来声明一个带一个或多个方法的接口,也可以传递参数和返回值。 由于远程调用的需要, 这些参数和返回值并不是任何类型.下面是些AIDL支持的数据类型:
1. 不需要import声明的简单Java编程语言类型(int,boolean等)
2. String, CharSequence不需要特殊声明
3. List, Map和Parcelables类型, 这些类型内所包含的数据成员也只能是简单数据类型, String等其他比支持的类型.
(另外: 我没尝试Parcelables, 在Eclipse+ADT下编译不过, 或许以后会有所支持)
AIDL(Android Interface Definition Language:接口描述语言)
跨进程访问(AIDL服务)
Android系统中的进程之间不能共享内存,因此,需要提供一些机制在不同进程之间进行数据通信。 我们知道4个Android应用程序组件中的3个(Activity、Broadcast和Content Provider)都可以进行跨进程访问,另外一个Android应用程序组件Service同样可以。这就是本节要介绍的AIDL服务。
什么是AIDL服务
为了使其他的应用程序也可以访问本应用程序提供的服务,Android系统采用了远程过程调用(Remote Procedure Call,RPC)方式来实现。与很多其他的基于RPC的解决方案一样,Android使用一种接口定义语言(Interface Definition Language,IDL)来公开服务的接口。因此,可以将这种可以跨进程访问的服务称为AIDL(Android Interface Definition Language)服务。
建立AIDL服务的步骤
建立AIDL服务要比建立普通的服务复杂一些,具体步骤如下:
(1) 在Eclipse Android工程的Java包目录中建立一个扩展名为aidl的文件。该文件的语法类似于Java代码,但会稍有不同。详细介绍见实例的内容。
(2) 如果aidl文件的内容是正确的,ADT会自动生成一个Java接口文件(*.java)。
(3)建立一个服务类(Service的子类)。
(4)实现由aidl文件生成的Java接口。
(5) 在AndroidManifest.xml文件中配置AIDL服务,尤其要注意的是,<action>标签中android:name的属性值就是客户端要引用该服务的ID,也就是Intent类的参数值。
第一个DEMO:有服务器端和客户端
实例:
服务器端:
本例中将建立一个简单的AIDL服务。这个AIDL服务只有一个getValue方法,该方法返回一个String类型的值。在安装完服务后,会在客户端调用这个getValue方法,并将返回值在TextView组件中输出。建立这个AIDL服务的步骤如下:
(1)建立一个aidl文件。在Java包目录中建立一个IMyService.aidl文件。IMyService.aidl文件的位置如图所示。
IMyService.aidl文件的内容如下:
package cn.m9.mobile.aidl;
interface IMyService {
String getValue();
}
IMyService.aidl文件的内容与Java代码非常相似,但要注意,不能加修饰符(例如,public、private)、AIDL服务不支持的数据类型(例如,InputStream、OutputStream)等内容。
(2)如果IMyService.aidl文件中的内容输入正确,ADT会自动生成一个IMyService.java文件。读者一般并不需要关心这个文件的具体内容,也不需要维护这个文件。
(3)编写一个MyService类。MyService是Service的子类,在MyService类中定义了一个内嵌类(MyServiceImpl),该类是IMyService.Stub的子类。MyService类的代码如下:
package cn.m9.mobile.aidl;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
public class MyService extends Service {
public class MyServiceImpl extends IMyService.Stub {
public String getValue()
{
return "Android is very powerful";
}
}
@Override
public IBinder onBind(Intent arg0) {
// TODO Auto-generated method stub
return new MyServiceImpl();
}
}
在编写上面代码时要注意如下两点:
IMyService.Stub是根据IMyService.aidl文件自动生成的,一般并不需要管这个类的内容,只需要编写一个继承于IMyService.Stub类的子类(MyServiceImpl类)即可。
onBind方法必须返回MyServiceImpl类的对象实例,否则客户端无法获得服务对象。
(4)在AndroidManifest.xml文件中配置MyService类,代码如下:
<service android:name=".MyService">
<intent-filter>
<action android:name="cn.m9.mobile.aidl.IMyService" />
</intent-filter>
</service>
其中"cn.m9.activity.IMyService"是客户端用于访问AIDL服务的ID。
下面来编写客户端的调用代码。
首先新建一个Eclipse Android工程(TestAidlClient),并将自动生成的IMyService.java文件连同包目录一起复制到testAIDL_client工程的src目录中(R文件除外),如图8.25所示。
调用AIDL服务首先要绑定服务,然后才能获得服务对象,代码如下:
package cn.m9.mobile;
import cn.m9.mobile.R;
import cn.m9.mobile.aidl.IMyService;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
public class TestAidlClient extends Activity implements OnClickListener {
/** Called when the activity is first created. */
private IMyService myService = null;
private Button btnInvokeAIDLService;
private Button btnBindAIDLService;
private TextView textView;
private static final String TAG="TestAIDLClient";
private ServiceConnection serviceConnection = new ServiceConnection()
{
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub
myService = IMyService.Stub.asInterface(service);
btnInvokeAIDLService.setEnabled(true);
}
@Override
public void onServiceDisconnected(ComponentName arg0) {
// TODO Auto-generated method stub
}
};
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
btnInvokeAIDLService = (Button) findViewById(R.id.btnInvokeAIDLService);
btnBindAIDLService = (Button) findViewById(R.id.btnBindAIDLService);
btnInvokeAIDLService.setEnabled(false);
textView = (TextView) findViewById(R.id.textview);
btnInvokeAIDLService.setOnClickListener(this);
btnBindAIDLService.setOnClickListener(this);
}
@Override
public void onClick(View view) {
// TODO Auto-generated method stub
switch (view.getId())
{
case R.id.btnBindAIDLService:
//绑定AIDL服务
bindService(new Intent("cn.m9.mobile.aidl.IMyService"), serviceConnection,Context.BIND_AUTO_CREATE);
break;
case R.id.btnInvokeAIDLService:
try{
textView.setText(myService.getValue());
} catch (Exception e)
{
}
break;
}
}
}
在编写上面代码时应注意如下两点:
使用bindService方法来绑定AIDL服务。其中需要使用Intent对象指定AIDL服务的ID,也就是<action>标签中android:name属性的值。
在绑定时需要一个ServiceConnection对象。创建ServiceConnection对象的过程中如果绑定成功,系统会调用onServiceConnected方法,通过该方法的service参数值可获得AIDL服务对象。
首先运行AIDL服务程序,然后运行客户端程序,单击【绑定AIDL服务】按钮,如果绑定成功,【调用AIDL服务】按钮会变为可选状态,单击这个按钮,会输出getValue方法的返回值,如下图所示:
第二个DEMO:有服务器端和客户端
既然AIDL是既然是可以在不同进程间进行操作,那么我们首先就需要准备两个项目,我们先来看看不同的两个项目结构式什么样子的。如图:
我们先暂且定义一个项目为“服务器端项目”,一个为“客户端项目”,其中紫色线框里面的东西是第一个需要注意的地方,就是不管你两个项目包名有多大差异,但是aidl文件必须放置在同一个包下面,而且名字也要相同。
我们现在创建好了项目,下一步就是开发调用接口和实现方法了。其实aidl文件在这里就充当的是一个interface类,你可以在它里面写上你要公布给客户端的调用接口。(具体代码我就不写了,大家可以看我上传的项目)在写aidl接口时,需要注意的是:类命必须和你的aidl文件名相同;必须指明当前包路径;如果是常用类型,如:String、List、int、void之类的,就不需要导入所在包路径了,当使用了外部类时,就需要导入这个类所在的包路径了;书写方法时,每个参数必须指定in或out来表明它是输入参数还是输出参数。
既然aidl我们写完了,那么接下来当然是要去实现这个接口,让它来帮我们做事情了, Aidl因为是远程调用,所以这里你的实现类必须是一个继承Service的类,然后在这个实现类中再去做aidl的实现方法。上图中 TestService就是我的实现类,具体代码也请看我上传的项目。不过在TestService需要注意的就是,onBind方法一定要返回你实现的 aidl实例对象,供远程调用方使用。
public IBinder onBind(Intent intent) {
return binder;
}
服务端manifest.xml中需要进行如下配置,不然你客户端调用会无法获得你的aidl对象的,我之前做测试的时候,一直被这个问题困扰。
<service android:name="TestService" android:process=":remote">
<intent-filter>配置你的AIDL接口完整路径
<action android:name="com.lifeblood.ITestService" />
</intent-filter>
</service>
做完了服务器的工作后,我们再开始做客户端的工作。客户端其实就很简单了,把服务器的aidl拷贝过来,创建一个ServiceConnection用来得到远程onBind中传递过来的aidl接口对象,然后在启动的时候进行一次绑定bindService就可以开始使用aidl中的方法了。
如果各位以后需要使用到aidl,那么只要注意以上说的,我想就不会出现什么大问题
第三个DEMO:
上面的两个示例代码都是有服务器端和客户端,那么这个示例只有一个应用程序,就是说这个应用既有服务器端,又有客户端,写到了一个应用里。当然,AIDL并不是说非得有两个应用才可以的,所以,大家仍然可以看看这个,是如何写到一个应用里的。
具体就不说明了,大家自己看源码更快些。
参考:
1. http://www.eoeandroid.com/forum- ... highlight-aidl.html
2. http://www.eoeandroid.com/forum- ... highlight-aidl.html
3. http://www.eoeandroid.com/forum- ... highlight-aidl.html
结尾:EOE有几个连载的文章,把链接贴出来,总结的非常好了,非常非常好。大家自己去看看吧。如果你说,这么多,我怎么看的过来的,我就想知道是怎么回事就行了,简单的看看代码,运行一下就可以了,那你就看我写的就够了。这么多,确实看着得需要点耐心。我们不是专家,我们不用研究那么多。
第一系列:
系列之Android AIDL, Parcelable和远程服务(一)的帖子链接http://www.eoeandroid.com/thread-91477-1-1.html
系列之Android AIDL, Parcelable和远程服务(二)的帖子链接http://www.eoeandroid.com/thread-91481-1-1.html
系列之Android AIDL, Parcelable和远程服务(三)的帖子链接http://www.eoeandroid.com/thread-91483-1-1.html
系列之Android AIDL, Parcelable和远程服务(四)http://www.eoeandroid.com/forum- ... fromuid-511991.html
第二系列:
系列之Android AIDL必看内容(一)的帖子链http://www.eoeandroid.com/thread-93384-1-1.html
系列之Android AIDL必看内容(二)的帖子链http://www.eoeandroid.com/thread-93387-1-1.html
系列之Android AIDL必看内容(三)的帖子链http://www.eoeandroid.com/thread-93388-1-1.html
系列之Android AIDL必看内容(四)的帖子链http://www.eoeandroid.com/thread-93878-1-1.html
系列之Android AIDL必看内容(五)的帖子链http://www.eoeandroid.com/thread-95062-1-1.html
系列之Android AIDL必看内容(六)http://www.eoeandroid.com/forum- ... fromuid-511991.html
第三系列:
系列之Android AIDL与传递对象(一)http://www.eoeandroid.com/forum- ... fromuid-511991.html
系列之Android AIDL与传递对象(二)的帖子链http://www.eoeandroid.com/thread-95088-1-1.html
系列之Android AIDL与传递对象(三)的帖子链http://www.eoeandroid.com/thread-95090-1-1.html
第四系列:
系列之Android Service介绍之AIDL(一)http://www.eoeandroid.com/forum- ... fromuid-511991.html
系列之Android Service介绍之AIDL(二)的帖子链http://www.eoeandroid.com/thread-95082-1-1.html
系列之Android Service介绍之AIDL(三)的帖子链http://www.eoeandroid.com/thread-95086-1-1.html
下面附上源码:
第一个DEMO:无
第二个DEMO: demo2.rar (68.68 KB, 下载次数: 0)
第三个DEMO: demo3.rar (51.66 KB, 下载次数: 0) |
|