Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 697|回复: 0

开发交流:面试题系列之二AIDL

[复制链接]

该用户从未签到

发表于 2011-10-24 09:31:00 | 显示全部楼层 |阅读模式
     
    这两天在看面试题,上面是这样问的: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)
回复

使用道具 举报

该用户从未签到

发表于 2011-10-24 09:31:02 | 显示全部楼层

Re:开发交

回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

GMT+8, 2025-1-10 19:32 , Processed in 0.325872 second(s), 36 queries .

Powered by Discuz! X3.4

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表