当前位置: 首页 > 编程日记 > 正文

android AIDL IPC深入分析

深入分析AIDL原理

  • 博客分类:
  • Android

在上一篇文章(Service使用方式)中,介绍了Android进程间通信(IPC)的使用,并给出了一个示例。但并没有深入分析aidl是怎样可以做到进程间通信的,它的执行过程是怎样的?

这篇文章来分析IRemoteService.aidl的执行过程,并理解aidl是怎样跨进程通信的。

当我们创建IRemoteService.aidl文件时,IDE会为我们在gen目录中创建相应的文件。

Iremoteservice.java代码  收藏代码
  1. /** This file is auto-generated.  DO NOT MODIFY.
  2. * Original file: F:\\workspace\\AndroidImprove\\src\\com\\example\\aidl\\IRemoteService.aidl
  3. */
  4. package com.example.aidl;
  5. public interface IRemoteService extends android.os.IInterface
  6. {
  7. /** Local-side IPC implementation stub class. */
  8. public static abstract class Stub extends android.os.Binder implements com.example.aidl.IRemoteService
  9. {
  10. private static final java.lang.String DESCRIPTOR = "com.example.aidl.IRemoteService";
  11. /** Construct the stub at attach it to the interface. */
  12. public Stub()
  13. {
  14. this.attachInterface(this, DESCRIPTOR);
  15. }
  16. /**
  17. * Cast an IBinder object into an com.example.aidl.IRemoteService interface,
  18. * generating a proxy if needed.
  19. */
  20. public static com.example.aidl.IRemoteService asInterface(android.os.IBinder obj)
  21. {
  22. if ((obj==null)) {
  23. return null;
  24. }
  25. android.os.IInterface iin = (android.os.IInterface)obj.queryLocalInterface(DESCRIPTOR);
  26. if (((iin!=null)&&(iin instanceof com.example.aidl.IRemoteService))) {
  27. return ((com.example.aidl.IRemoteService)iin);
  28. }
  29. return new com.example.aidl.IRemoteService.Stub.Proxy(obj);
  30. }
  31. public android.os.IBinder asBinder()
  32. {
  33. return this;
  34. }
  35. @Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException
  36. {
  37. switch (code)
  38. {
  39. case INTERFACE_TRANSACTION:
  40. {
  41. reply.writeString(DESCRIPTOR);
  42. return true;
  43. }
  44. case TRANSACTION_register:
  45. {
  46. data.enforceInterface(DESCRIPTOR);
  47. com.example.aidl.IRemoteCallback _arg0;
  48. _arg0 = com.example.aidl.IRemoteCallback.Stub.asInterface(data.readStrongBinder());
  49. this.register(_arg0);
  50. reply.writeNoException();
  51. return true;
  52. }
  53. case TRANSACTION_unregister:
  54. {
  55. data.enforceInterface(DESCRIPTOR);
  56. com.example.aidl.IRemoteCallback _arg0;
  57. _arg0 = com.example.aidl.IRemoteCallback.Stub.asInterface(data.readStrongBinder());
  58. this.unregister(_arg0);
  59. reply.writeNoException();
  60. return true;
  61. }
  62. case TRANSACTION_execute:
  63. {
  64. data.enforceInterface(DESCRIPTOR);
  65. this.execute();
  66. reply.writeNoException();
  67. return true;
  68. }
  69. case TRANSACTION_getStatus:
  70. {
  71. data.enforceInterface(DESCRIPTOR);
  72. java.lang.String _arg0;
  73. _arg0 = data.readString();
  74. int _result = this.getStatus(_arg0);
  75. reply.writeNoException();
  76. reply.writeInt(_result);
  77. return true;
  78. }
  79. }
  80. return super.onTransact(code, data, reply, flags);
  81. }
  82. private static class Proxy implements com.example.aidl.IRemoteService
  83. {
  84. private android.os.IBinder mRemote;
  85. Proxy(android.os.IBinder remote)
  86. {
  87. mRemote = remote;
  88. }
  89. public android.os.IBinder asBinder()
  90. {
  91. return mRemote;
  92. }
  93. public java.lang.String getInterfaceDescriptor()
  94. {
  95. return DESCRIPTOR;
  96. }
  97. //注册回调
  98. public void register(com.example.aidl.IRemoteCallback callback) throws android.os.RemoteException
  99. {
  100. android.os.Parcel _data = android.os.Parcel.obtain();
  101. android.os.Parcel _reply = android.os.Parcel.obtain();
  102. try {
  103. _data.writeInterfaceToken(DESCRIPTOR);
  104. _data.writeStrongBinder((((callback!=null))?(callback.asBinder()):(null)));
  105. mRemote.transact(Stub.TRANSACTION_register, _data, _reply, 0);
  106. _reply.readException();
  107. }
  108. finally {
  109. _reply.recycle();
  110. _data.recycle();
  111. }
  112. }
  113. //取消注册回调
  114. public void unregister(com.example.aidl.IRemoteCallback callback) throws android.os.RemoteException
  115. {
  116. android.os.Parcel _data = android.os.Parcel.obtain();
  117. android.os.Parcel _reply = android.os.Parcel.obtain();
  118. try {
  119. _data.writeInterfaceToken(DESCRIPTOR);
  120. _data.writeStrongBinder((((callback!=null))?(callback.asBinder()):(null)));
  121. mRemote.transact(Stub.TRANSACTION_unregister, _data, _reply, 0);
  122. _reply.readException();
  123. }
  124. finally {
  125. _reply.recycle();
  126. _data.recycle();
  127. }
  128. }
  129. //执行回调
  130. public void execute() throws android.os.RemoteException
  131. {
  132. android.os.Parcel _data = android.os.Parcel.obtain();
  133. android.os.Parcel _reply = android.os.Parcel.obtain();
  134. try {
  135. _data.writeInterfaceToken(DESCRIPTOR);
  136. mRemote.transact(Stub.TRANSACTION_execute, _data, _reply, 0);
  137. _reply.readException();
  138. }
  139. finally {
  140. _reply.recycle();
  141. _data.recycle();
  142. }
  143. }
  144. //获取状态
  145. public int getStatus(java.lang.String flag) throws android.os.RemoteException
  146. {
  147. android.os.Parcel _data = android.os.Parcel.obtain();
  148. android.os.Parcel _reply = android.os.Parcel.obtain();
  149. int _result;
  150. try {
  151. _data.writeInterfaceToken(DESCRIPTOR);
  152. _data.writeString(flag);
  153. mRemote.transact(Stub.TRANSACTION_getStatus, _data, _reply, 0);
  154. _reply.readException();
  155. _result = _reply.readInt();
  156. }
  157. finally {
  158. _reply.recycle();
  159. _data.recycle();
  160. }
  161. return _result;
  162. }
  163. }
  164. static final int TRANSACTION_register = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
  165. static final int TRANSACTION_unregister = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1);
  166. static final int TRANSACTION_execute = (android.os.IBinder.FIRST_CALL_TRANSACTION + 2);
  167. static final int TRANSACTION_getStatus = (android.os.IBinder.FIRST_CALL_TRANSACTION + 3);
  168. }
  169. //注册回调
  170. public void register(com.example.aidl.IRemoteCallback callback) throws android.os.RemoteException;
  171. //取消注册回调
  172. public void unregister(com.example.aidl.IRemoteCallback callback) throws android.os.RemoteException;
  173. //执行回调
  174. public void execute() throws android.os.RemoteException;
  175. //获取状态
  176. public int getStatus(java.lang.String flag) throws android.os.RemoteException;
  177. }

在ClientActivity绑定远程Service并建立连接时会调用ServiceConnection.onServiceConnected(ComponentName name, IBinder service)

Java代码  收藏代码
  1. public void onServiceConnected(ComponentName name, IBinder service) {
  2. remoteService = IRemoteService.Stub.asInterface(service);
  3. //注册回调
  4. try {
  5. remoteService.register(remoteCallback);
  6. catch (RemoteException e) {
  7. e.printStackTrace();
  8. }
  9. }

IBinder service是从RemoteService返回的IRemoteService.Stub iBinder,这个对象是Server应用进程中的对象。

IRemoteService.Stub.asInterface(service)在本地创建了一个代理

public static com.example.aidl.IRemoteService asInterface(android.os.IBinder obj)

{

if ((obj==null)) {

return null;

}

android.os.IInterface iin = (android.os.IInterface)obj.queryLocalInterface(DESCRIPTOR);//这里肯定返回null

if (((iin!=null)&&(iin instanceof com.example.aidl.IRemoteService))) {

return ((com.example.aidl.IRemoteService)iin);

}

return new com.example.aidl.IRemoteService.Stub.Proxy(obj);//创建一个本地代理

}

当使用remoteService调用方法时,其实是调用了本地com.example.aidl.IRemoteService.Stub.Proxy对象的方法,从Proxy方法中可以看到,每个方法都执行了mRemote.transact(Stub.TRANSACTION_xxx, _data, _reply, 0);。

如:

Java代码  收藏代码
  1. //获取状态
  2. public int getStatus(java.lang.String flag) throws android.os.RemoteException
  3. {
  4. android.os.Parcel _data = android.os.Parcel.obtain();
  5. android.os.Parcel _reply = android.os.Parcel.obtain();
  6. int _result;
  7. try {
  8. _data.writeInterfaceToken(DESCRIPTOR);
  9. _data.writeString(flag);
  10. mRemote.transact(Stub.TRANSACTION_getStatus, _data, _reply, 0);
  11. _reply.readException();
  12. _result = _reply.readInt();
  13. }
  14. finally {
  15. _reply.recycle();
  16. _data.recycle();
  17. }
  18. return _result;
  19. }

这一过程是把Client端的参数转换成Parcel(_data)传递到Server端,而在Server端又会把返回数据保存到_reply中,这就形成了一次交互。

mRemote是远程对象,transact方法会执行onTransact方法

Java代码  收藏代码
  1. public final boolean transact(int code, Parcel data, Parcel reply,
  2. int flags) throws RemoteException {
  3. if (Config.LOGV) Log.v("Binder""Transact: " + code + " to " + this);
  4. if (data != null) {
  5. data.setDataPosition(0);
  6. }
  7. boolean r = onTransact(code, data, reply, flags);
  8. if (reply != null) {
  9. reply.setDataPosition(0);
  10. }
  11. return r;
  12. }

这样就会执行远程的onTransact方法,

Java代码  收藏代码
  1. case TRANSACTION_getStatus:
  2. {
  3. data.enforceInterface(DESCRIPTOR);
  4. java.lang.String _arg0;
  5. _arg0 = data.readString();
  6. int _result = this.getStatus(_arg0);
  7. reply.writeNoException();
  8. reply.writeInt(_result);
  9. return true;
  10. }

注意 int _result = this.getStatus(_arg0);,这就调用了Server端的getStatus(String flag),并把返回结果写到Client端的代理Proxy对象的_reply中

到此,aidl通信过程就完成了。

PS: aidl通信有点复杂,但仔细分析并不是很难

相关文章:

xml常用操作(js、sql、vb)

我们经常会用到xml操作,如下介绍了js、sql、vb等对xml的操作。 JS创建xml对象 //创建对象 function getDataXML() {var objTds $("TEXTAREA");var count objTds.length;var jsonData;var xmlDoc new ActiveXObject("Microsoft.XMLDOM");//创…

【ACM】杭电OJ 2019

可以用链表写&#xff0c;也可以用顺序表写&#xff1a; 下面是链表的AC代码&#xff1a; #include <stdio.h> #include <stdlib.h> typedef struct node {int info;struct node* next; }N; N* init() {return NULL; } N* creat(N* head,int n) {int i,x;N *p, *…

Nancy in .Net Core学习笔记 - 初识Nancy

原文:Nancy in .Net Core学习笔记 - 初识Nancy前言 去年11月份参加了青岛MVP线下活动&#xff0c;会上老MVP衣明志介绍了Nancy, 一直没有系统的学习一下&#xff0c;最近正好有空&#xff0c;就结合.NET Core学习总结了一下。 注&#xff1a; 本文中大部分内容都是对官网文档的…

rpath和runpath的区别

考虑翻译Qt官方blog中的RPATH and RUNPATH这篇文章&#xff0c;在继续之前&#xff0c;我需要先验证自己的理解是正确的&#xff0c;至少能自圆其说&#xff0c;能说服自己。 用例子说话 二进制 对应源码 有一个程序 a.out main.c 需要加载插件A libA.so liba…

scanf(%s,a)和gets(a)的差别

gets()和scanf()的区别在于输入的字符串是否中间有空格&#xff1a;对于前者&#xff0c;只有遇到"\n"时才停止输入&#xff0c;而对于后者&#xff0c;出现"\n"或空格都停止输入。

阿里注册中心nacos使用整合Dubbo-原创

阿里注册中心nacos是今年开源的框架&#xff0c;一开始以为就是个zk。后面看了图才明白他对标的竟然是consul\eureka&#xff0c;最重要是完美支持dubbo。我想今年开源它也是别有用意 。&#xff08;目前nacos0.7版本&#xff09; Dubbo 融合 Nacos 成为注册中心 Nacos 作为 Du…

UBUNTU adb连接android设备

1sudo vi /etc/udev/rules.d/70-android.rules 2最新修改方法&#xff0c;不用去看设备的ID&#xff0c;直接在rules.d下增加一个文件51-android.rules&#xff0c;内容为&#xff1a; SUBSYSTEM"usb" ENV{DEVTYPE}"usb_device", MODE"0666" …

jetty9请求form表单太小限制

报错&#xff1a;java.lang.IllegalStateException: Form too large: 201975 > 200000解决&#xff1a;vi jetty.xml<Configure id"Server" class"org.eclipse.jetty.server.Server">在Server这行下面增加以下代码<!-- guowang add --><…

【ACM】杭电OJ 2037

题目链接&#xff1a;杭电OJ 2037 先把b[i]进行排序&#xff0c;然后&#xff0c;b[i]与a[i1]进行比较。 #include <iostream> #include <cstdio> #include <cstring> #include <cmath> #include <cstdlib> #include <set> #include <…

第二十章:异步和文件I/O.(十三)

通过该开销&#xff0c;可以开始实际编写应用程序。 TextFileAsyncPage的XAML文件与TextFileTryoutPage相同&#xff0c;但必须将代码隐藏文件设置为使用异步文件I / O方法。 必须在此处捕获文件I / O函数中可能发生的任何异常&#xff0c;这意味着任何可以抛出异常的方法必须与…

shell [] [[ ]] {}区别

一、小括号&#xff0c;圆括号&#xff08;&#xff09; 1、单小括号 () ①命令组。括号中的命令将会新开一个子shell顺序执行&#xff0c;所以括号中的变量不能够被脚本余下的部分使用。括号中多个命令之间用分号隔开&#xff0c;最后一个命令可以没有分号&#xff0c;各命…

【C++】用指针做函数参数

此篇博客程序运行环境为&#xff1a;VS2017&#xff01;&#xff01;&#xff01; 函数的参数不仅可以是整型、浮点型、字符型等数据&#xff0c;还可以是指针类型。 它的作用是将一个变量的的地址传给被调用函数的形参。 e.g. 输入两个数&#xff0c;按由大到小顺序输出 #…

JAVA面向对象-----final关键字

JAVA面向对象—–final关键字 1&#xff1a;定义静态方法求圆的面积 2&#xff1a;定义静态方法求圆的周长 3&#xff1a;发现方法中有重复的代码&#xff0c;就是PI&#xff0c;圆周率。1&#xff1a;如果需要提高计算精度&#xff0c;就需要修改每个方法中圆周率。 4&#xf…

win7安装mysql-8.0.13-winx64

这里展示一下&#xff0c;由于需要安装一个版本测试一下数据&#xff0c;其实就是超简单的啦。 下包 注:https://dev.mysql.com/downloads/mysql/ 解压与配置 [mysqld] basedirC:\\Users\\hp\\Downloads\\mysql-8.0.13-winx64 datadirC:\\Users\\hp\\Downloads\\mysql-8.0.13-w…

Http 请求头中的 Proxy-Connection

平时用 Chrome 开发者工具抓包时&#xff0c;经常会见到 Proxy-Connection 这个请求头。之前一直没去了解什么情况下会产生它&#xff0c;也没去了解它有什么含义。最近看完《HTTP 权威指南》第四章「连接管理」和第六章「代理」之后&#xff0c;终于搞明白了这是因为给浏览器设…

【C++】枚举类型

如果一个变量只能有几种可能的值&#xff0c;可以定义为枚举类型。所谓“枚举”就是把变量的值一一列出来&#xff0c;变量的值只能在列出来的值的范围内。 声明枚举类型的一般形式&#xff1a; enum 枚举类型名 {枚举常量表} enum weekday {sun,mon,tue,wed,thu,fri,sat}; …

ubuntu设置securecrt串口权限

在普通用户的模式下&#xff0c;用SecureCRT链接串口交换机&#xff0c;开始会提示/dev/ttyUSB0权限不足&#xff0c;无法打开&#xff0c;临时的解决办法是 chmod 0rw /dev/ttyUSB0 但是这个重启后便没了作用&#xff0c;下面的方法能在重启后让普通用户链接串口设备。 sudo v…

深入解析Angular Component的源码示例

本篇文章主要介绍了剖析Angular Component的源码示例&#xff0c;写的十分的全面细致&#xff0c;具有一定的参考价值&#xff0c;对此有需要的朋友可以参考学习下。如有不足之处&#xff0c;欢迎批评指正。 Web Component 定义 W3C为统一组件化标准方式&#xff0c;提出Web Co…

VS2017 cout 不明确

各种头文件没问题。直接声明名称空间 using namespace std&#xff1b; 解决方法&#xff1a; 然后把using namespace std;这句给注释掉&#xff0c;等出现错误提示&#xff0c;在取消注释&#xff0c;然后就好了

google breakpad native crash分析工具

一. BreakPad简介Google breakpad是一个跨平台的崩溃转储和分析框架和工具集合。Breakpad由三个主要组件&#xff1a;client&#xff0c;以library的形式内置在你的应用中&#xff0c;当崩溃发生时写 minidump文件symbol dumper, 读取由编译器生成的调试信息&#xff08;debugg…

Java基础教程(15)--枚举类型

枚举类型定义了一个枚举值的列表&#xff0c;每个值是一个标识符。例如&#xff0c;下面的语句声明了一个枚举类型&#xff0c;用来表示星期的可能情况&#xff1a; public enum Day {SUNDAY, MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY } 实际上&#xff0c;这个…

基于AOA协议的android USB通信

摘 要&#xff1a;AOA协议是Google公司推出的用于实现Android设备与外围设备之间USB通信的协议。该协议拓展了Android设备USB接口的功能&#xff0c;为基于Android系统的智能设备应用于数据采集和设备控制领域提供了条件。介绍了Android系统下USB通信的两种模式&#xff0c;并给…

Linux下使用ssh动态验证码登陆机器

ssh动态验证码登录机器Google Authenticator是一个动态验证码程序&#xff0c;兼容各种智能手机平板设备&#xff0c;可以用来做各种帐号的二次验证&#xff0c;增加帐号的安全性。SSH是Linux系统的最重要防线之一&#xff0c;为了防止密码泄露或者被爆破&#xff0c;可以使用G…

【C++】枚举类型应用

运行环境&#xff1a;VS2017 可以参考&#xff1a;【C】枚举类型 医院内科有A&#xff0c;B&#xff0c;C&#xff0c;D&#xff0c;E&#xff0c;F&#xff0c;G共七位医生&#xff0c;每人在一周内要值一次夜班&#xff0c;排班的要求&#xff1a; &#xff08;1&#xff…

量子力学又一突破,中国科学家首次实现量子纠缠态自检验

这也是国际上首个具有“高可靠、抗干扰”特性的纠缠态自检验实验。 最近&#xff0c;量子力学领域又传来好消息&#xff0c;中国科学技术大学的郭光灿院士团队在实验中首次实现了量子纠缠态的自检验&#xff0c;推动了自检验在各种量子信息过程中的基础发展。 何为量子纠缠&a…

awk命令中执行多条shell命令

awk中使用的shell命令&#xff0c;有2种方法&#xff1a;一。使用system&#xff08;&#xff09;二。使用print cmd | “/bin/bash”http://www.gnu.org/software/gawk/manual/gawk.html#I_002fO-Functions一。使用所以system&#xff08;&#xff09;awk程序中我们可以使用sy…

LAMP高级环境实战

LAMP架构应用实战介绍LAMP指的Linux&#xff08;操作系统&#xff09;、Apache&#xff08;HTTP 服务器&#xff09;&#xff0c;MySQL&#xff08;数据库软件&#xff09; 和PHP&#xff08;有时也是指Perl或Python&#xff09; 的第一个字母&#xff0c;一般用来建立web 服务…

【C++】用类来处理排序问题

运行环境&#xff1a;VS2017 由小到大排序 可以看出在主函数中所做的事&#xff1a; &#xff08;1&#xff09;定义对象。 &#xff08;2&#xff09;向各对象发出“消息”&#xff0c;通知各对象完成有关任务。即调用有关对象的成员函数&#xff0c;去完成相应的操作。 …

winform 弹出窗体位置设定

[转]https://www.cnblogs.com/liushenglin/p/5350641.html 一、C#中弹出窗口位置 加入命名空间using System.Drawing和using System.Windows.Forms假定窗口名为form1,则 form1.StartPosition FormStartPosition.CenterScreen;窗体位置在屏幕中间form1.StartPosition FormSta…

pkg-config工具在实际工程中的用法

在如今这个开源的环境里&#xff0c;想要开发某个功能&#xff0c;我们都会下意识的上网搜索有没有开源库&#xff0c;如果有开源库&#xff0c;那么好&#xff0c;下载下来给它编译好&#xff0c;使用。但是在使用过程中&#xff0c;你是否遇到不知如何将第三方库编译&#xf…