Android AIDL实例解析

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介:

AIDL这项技术在我们的开发中一般来说并不是很常用,虽然自己也使用新浪微博的SSO登录,其原理就是使用AIDL,但是自己一直没有动手完整的写过AIDL的例子,所以就有了这篇简单的文章。
AIDL(AndRoid接口描述语言)是一种借口描述语言; 编译器可以通过aidl文件生成一段代码,通过预先定义的接口达到两个进程内部通信进程的目的. 如果需要在一个Activity中, 访问另一个Service中的某个对象, 需要先将对象转化成 AIDL可识别的参数(可能是多个参数), 然后使用AIDL来传递这些参数, 在消息的接收端, 使用这些参数组装成自己需要的对象.
说白了,AIDL就是定义一个接口,客户端(调用端)通过bindService来与远程服务端简历一个连接,在该连接建立时会将返回一个IBinder对象,该对象是服务端Binder的BinderProxy,在建立连接时,客户端通过asInterface函数将该BinderProxy对象包装成本地的Proxy,并将远程服务端的BinderProxy对象赋值给Proxy类的mRemote字段,就是通过mRemote执行远程方法调用。需要对Binder机制有更深的理解,请转到老罗的系列文字吧,Android系统进程间通信Binder机制在应用程序框架层的Java接口源代码分析。下面我们看一个AIDL实例。


AIDL接口声明
在src目录下创建一个com.example.advanceandroid.aidl包,然后在该包下创建一个ILogin.aidl文件,注意是创建文件而不是类或者接口类型。在ILogin.aidl中声明接口,实例如下 :

package com.example.advanceandroid.aidl;

interface ILogin {
String login();
}

注意看,接口和方法声明都不用public,方法加入public会提示错误。编写完后如果eclipse开启了自动编译则会在gen/com.example.advanceandroid.aidl下生成一个ILogin.java类,内容大致如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
package  com.example.advanceandroid.aidl;
 
public  interface  ILogin extends  android.os.IInterface
{
     /** Local-side IPC implementation stub class. */
     public  static  abstract  class  Stub extends  android.os.Binder implements
             com.example.advanceandroid.aidl.ILogin
     {
         private  static  final  java.lang.String DESCRIPTOR = "com.example.advanceandroid.aidl.ILogin" ;
 
         /** Construct the stub at attach it to the interface. */
         public  Stub()
         {
             this .attachInterface( this , DESCRIPTOR);
         }
 
         /**
          * Cast an IBinder object into an com.example.advanceandroid.aidl.ILogin
          * interface, generating a proxy if needed.
          */
         public  static  com.example.advanceandroid.aidl.ILogin asInterface(android.os.IBinder obj)
         {
             if  ((obj == null )) {
                 return  null ;
             }
             android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
             if  (((iin != null ) && (iin instanceof  com.example.advanceandroid.aidl.ILogin))) {
                 return  ((com.example.advanceandroid.aidl.ILogin) iin);
             }
             return  new  com.example.advanceandroid.aidl.ILogin.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_login: {                            // 1、登录请求,执行的是this.login();
                     data.enforceInterface(DESCRIPTOR);
                     java.lang.String _result = this .login();
                     reply.writeNoException();
                     reply.writeString(_result);
                     return  true ;
                 }
             }
             return  super .onTransact(code, data, reply, flags);
         }
 
         private  static  class  Proxy implements  com.example.advanceandroid.aidl.ILogin
         {
             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 login() throws  android.os.RemoteException                // 2、Proxy中的login,通过Binder机制实现IPC
             {
                 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_login, _data, _reply, 0 );
                     _reply.readException();
                     _result = _reply.readString();
                 } finally  {
                     _reply.recycle();
                     _data.recycle();
                 }
                 return  _result;
             }
         }
 
         static  final  int  TRANSACTION_login = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0 );
     }
 
     public  java.lang.String login() throws  android.os.RemoteException;
}

可以看到,该类中自动生成了ILogin接口,该接口中又一个login()函数。但最重要的是里面生成了一个Stub类,该类集成子Binder类,并且实现了ILogin接口。Stub里面最重要的就是asInterface()这个函数,在这个函数中会判断obj参数的类型,如果是该obj是本地的接口类似,则认为不是IPC,会将该obj转换成ILogin类型;否则会通过自动生成的另一个内部类Proxy来包装obj,将其赋值给Proxy中的mRemote属性。Proxy类也实现了ILogin接口,在login()函数中,Proxy将通过Binder机制向服务端传递请求和数据,如上面代码中的注释2。这是客户端的工作算是完成了。

服务端AIDL接口
服务端也需要在相同的包下创建同名的aidl文件,我们直接将客户端的com.example.advanceandroid.aidl包下的ILogin.aidl拷贝到服务端即可,如果用到了自定义的类型,那么该自定义类型也需要在客户端、服务端都有。拷贝完aidl后,在服务端程序中也会在gen中生成对应的ILogin.java文件,内容同客户端一样。这里的重点我们要看onTransact函数,即上述代码中的注释1处,可以看到,在case TRANSACTION_login处执行了this.login()函数,意思是当接收到客户端的TRANSACTION_login请求时,执行this.login()函数,通过客户端的分析我们知道,当我们调用login()时实际上就是通过mRemote向服务端提交了一个TRANSACTION_login请求,因此就两端通过Binder机制就对接上了,我们可以简单的理解为C/S模式。

服务端还没有完,最重要的一步时建立一个Service,内容大致如下 :

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/**
* AIDL服务端接口,LoginStubImpl实现了ILogin接口.
*
* @author mrsimple
*/
public  class  LoginService extends  Service {
 
     /**
      *
      */
     IBinder mBinder = new  LoginStubImpl();
 
     /**
      * @author mrsimple
      */
     class  LoginStubImpl extends  Stub {
         @Override
         public  String login() throws  RemoteException {
             return  "这是从 "  + this .getClass().getName() + " 返回的字符串" ;
         }
     }
 
     /*
      * 返回Binder实例,即实现了ILogin接口的Stub的子类,这里为LoginStubImpl
      * [url=home.php?mod=space&uid=133757]@see[/url] android.app.Service#onBind(android.content.Intent)
      */
     @Override
     public  IBinder onBind(Intent intent) {
         return  mBinder;
     }
}

该Service我们这里命名为LoginService,继承自Service,然后建一个名为LoginServiceImpl的内部类,该类继承自自动生成的Stub,然后实现login()方法。在LoginService中声明一个IBinder字段mBinder :

IBinder mBinder = new LoginStubImpl();

并且在LoginService的onBind函数中将mBinder对象返回。即在客户端建立与服务端的连接时,会调用onBind方法将mBinder对象返回,在客户端的ServiceConnection类的onServiceConnected函数中得到的对象IBinder就是经过BinderProxy包装的LoginService中的mBinder对象。因此在服务端中的onTransact中调用的this.login()函数实际上就是调用的LoginStubImpl中的login()函数。

在服务端程序的AndroidManifest.xml中注册LoginService,如下 :

1
2
3
4
5
6
<!-- aidl server service -->
     <service android:name= "com.example.advanceandroid.aidl.LoginService" >
         <intent-filter>
             
         </action></intent-filter>
     </service>


客户端建立连接

在Activity中加入如下代码 :

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
ServiceConnection mLoginConnection = new  ServiceConnection() {
 
        @Override
        public  void  onServiceDisconnected(ComponentName name) {
            Log.d( "" , "### aidl disconnected." );
        }
 
        @Override
        public  void  onServiceConnected(ComponentName name, IBinder service) {
            Log.d( "" , "### aidl onServiceConnected.     service : " + service.getClass().getName());
 
            ILogin login = Stub.asInterface(service);
            Log.d( "" , "### after asInterface : "  + login.getClass().getName());
            try  {
                Log.d( "" , "### login : "  + login.login());
                // Toast.makeText(MainActivity.this, "onServiceConnected : " +
                // login.login(),
                // Toast.LENGTH_SHORT).show();
            } catch  (RemoteException e) {
                e.printStackTrace();
            }
        }
    };
 
    @Override
    protected  void  onResume() {
        super .onResume();
 
        // 服务端的action
        Intent aidlIntent = new  Intent( "com.example.advanceandroid.aidl.LoginService" );
        bindService(aidlIntent, mLoginConnection, Context.BIND_AUTO_CREATE);
    }
 
    @Override
    protected  void  onStop() {
        super .onStop();
        // unbind
        unbindService(mLoginConnection);
    }

运行

先运行服务端程序,然后在启动客户端程序,可以看到客户端输出如下Log:

1
2
3
09 - 02  10 : 40 : 54.662 : D/( 9589 ): ### aidl onServiceConnected.     service : android.os.BinderProxy
09 - 02  10 : 40 : 54.662 : D/( 9589 ): ### after asInterface : com.example.advanceandroid.aidl.ILogin$Stub$Proxy
09 - 02  10 : 40 : 54.662 : D/( 9589 ): ### login : 这是从 com.example.advanceandroid.aidl.LoginService$LoginStubImpl 返回的字符串

可以看淡onServiceConnected(ComponentName name, IBinder service)中的service对象是BinderProxy类型,经过asInterface转换后被包装成了Proxy类型,但是调用的时候,执行的是服务端LoginStubImpl中的login()函数。因此,LoginStubImpl实例mBinder被服务端包装成BinderProxy类型,再经过客户端的Proxy进行包装,通过Binder机制进行数据传输,实现IPC。



    本文转自 一点点征服   博客园博客,原文链接:http://www.cnblogs.com/ldq2016/p/7306173.html,如需转载请自行联系原作者



相关文章
|
26天前
|
Java 开发工具 Android开发
Android与iOS开发环境搭建全解析####
本文深入探讨了Android与iOS两大移动操作系统的开发环境搭建流程,旨在为初学者及有一定基础的开发者提供详尽指南。我们将从开发工具的选择、环境配置到第一个简单应用的创建,一步步引导读者步入移动应用开发的殿堂。无论你是Android Studio的新手还是Xcode的探索者,本文都将为你扫清开发道路上的障碍,助你快速上手并享受跨平台移动开发的乐趣。 ####
|
2天前
|
数据挖掘 vr&ar C++
让UE自动运行Python脚本:实现与实例解析
本文介绍如何配置Unreal Engine(UE)以自动运行Python脚本,提高开发效率。通过安装Python、配置UE环境及使用第三方插件,实现Python与UE的集成。结合蓝图和C++示例,展示自动化任务处理、关卡生成及数据分析等应用场景。
17 5
|
19天前
|
存储 Linux API
深入探索Android系统架构:从内核到应用层的全面解析
本文旨在为读者提供一份详尽的Android系统架构分析,从底层的Linux内核到顶层的应用程序框架。我们将探讨Android系统的模块化设计、各层之间的交互机制以及它们如何共同协作以支持丰富多样的应用生态。通过本篇文章,开发者和爱好者可以更深入理解Android平台的工作原理,从而优化开发流程和提升应用性能。
|
19天前
|
存储 网络协议 算法
【C语言】进制转换无难事:二进制、十进制、八进制与十六进制的全解析与实例
进制转换是计算机编程中常见的操作。在C语言中,了解如何在不同进制之间转换数据对于处理和显示数据非常重要。本文将详细介绍如何在二进制、十进制、八进制和十六进制之间进行转换。
28 5
|
19天前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
26天前
|
存储 机器学习/深度学习 编解码
阿里云服务器计算型c8i实例解析:实例规格性能及使用场景和最新价格参考
计算型c8i实例作为阿里云服务器家族中的重要成员,以其卓越的计算性能、稳定的算力输出、强劲的I/O引擎以及芯片级的安全加固,广泛适用于机器学习推理、数据分析、批量计算、视频编码、游戏服务器前端、高性能科学和工程应用以及Web前端服务器等多种场景。本文将全面介绍阿里云服务器计算型c8i实例,从规格族特性、适用场景、详细规格指标、性能优势、实际应用案例,到最新的活动价格,以供大家参考。
|
26天前
|
开发框架 Dart Android开发
安卓与iOS的跨平台开发:Flutter框架深度解析
在移动应用开发的海洋中,Flutter作为一艘灵活的帆船,正引领着开发者们驶向跨平台开发的新纪元。本文将揭开Flutter神秘的面纱,从其架构到核心特性,再到实际应用案例,我们将一同探索这个由谷歌打造的开源UI工具包如何让安卓与iOS应用开发变得更加高效而统一。你将看到,借助Flutter,打造精美、高性能的应用不再是难题,而是变成了一场创造性的旅程。
|
28天前
|
安全 Java Linux
深入解析Android系统架构及其对开发者的意义####
【10月更文挑战第21天】 本文旨在为读者揭开Android操作系统架构的神秘面纱,探讨其如何塑造现代移动应用开发格局。通过剖析Linux内核、硬件抽象层、运行时环境及应用程序框架等关键组件,揭示Android平台的强大功能与灵活性。文章强调了理解Android架构对于开发者优化应用性能、提升用户体验的重要性,并展望了未来技术趋势下Android的发展方向。 ####
44 0
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
71 2
|
2月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
76 0

推荐镜像

更多
下一篇
DataWorks