理解android AIDL

理解Android AIDL

在研究了 Android Frameworks 中进程间通信(IPC)相关的一些程序后,了解到 Android 系统中进程间通信的机制绝大部分就是 Binder,主要表现在系统服务的调用,app进程间功能调用等。而 Android 上实现 Binder 的具体形式就是 AIDL (Android Interface Definition Language) 。

所以在去充分理解 Binder 之前,要先理解下 AIDL 。

AIDL概述

Android 接口定义语言 (AIDL) 是一种让用户抽象出 IPC 的工具。给定一个接口(在.aidl文件中指定),各种构建系统使用 aidl 二进制文件来构建 C++ 或 Java 绑定,以便该接口可以跨进程使用,而不管那里的运行时或位数如何。

AIDL 可以在 Android 中的任何进程之间使用:平台组件之间或应用程序之间。

下面是一个 AIDL 接口定义示例:

package my.package; // 与普通java文件定义一样有包名import my.package.Foo; // 可以是在其他地方定义的类型interface IFoo {void doFoo(Foo foo);
}

服务器 进程注册一个接口并为其提供调用服务,而 客户端 进程则调用这些接口。在一些情况下,一个 app 进程既充当客户端又充当服务器,因为它可能引用多个接口。

运行

AIDL 使用 binder 内核驱动调用。当进行一次调用时,方法标识符和所有的数据被打包进缓存,并将其拷贝到正在等待读取数据的 binder 线程的远程进程中。当 binder 线程因为业务需要接收数据时,binder 线程会在本地进程中找到本地存根对象,并且这个对象会解包数据并且调用本地接口对象(AIDL定义的接口)。这个本地接口对象是服务进程创建并注册。当调用是在同一个进程和同一个后端发生时,不会产生代理对象,这样的直接调用也不会有任何打包或解包操作。

AIDL 接口的调用是直接函数调用。调用线程实际情况的差异取决于调用是来自本地进程中的线程,还是远程进程中的线程。

从线程角度看 AIDL 接口调用:

  1. 来自本地进程的调用在发起调用的同一线程内执行。如果调用来自main线程,那么它将继续在 AIDL 接口中执行。如果线程是其他线程,那么其便是在服务中执行代码的线程。因此,若只是发生在service内的线程之间的互相访问,那么可以自行控制调用发生在哪个线程。在这种情况下,可以不定义aidl,直接定义一个接口实现 Binder

  2. 从远程进程中平台持有的线程池中的某个线程发起的调用。为来自未知线程以及同时发生多次调用的情况做好准备。换言之,AIDL 接口调用必须是线程安全的。来自远程进程同一线程内相同对象的多个调用会在同一接收端按顺序调用。

  3. oneway 关键字修饰的远程调用。调用端方法不会阻塞。方法会在发送完数据后立即返回。接口实现最终按常规方式从 Binder 线程池接收这个正常的远程调用。

AIDL 定义使用

跨进程的 aidl 调用。在服务端和客户端分别定义,调用相关的代码。

服务端定义 aidl 文件及接口。

  1. 使用 java 语法创建 .aidl文件。在包含有 .aidl 文件的每个 application 中进行构建,Android 的 aidl 工具会基于 .aidl 文件内容在 build/generated/aidl_source_output/dir 目录下生成对应的 .java 文件。

    从 Android 12 开始,要创建编译 aidl 文件,需要的 build.gradle 文件 buildFeatures 块中添加 aidl=true 的设置。 Android 11 之前可以直接编译 aidl 文件。

    Android 12 开始需要在 build.gradle 中添加 aidl 构建配置项。若 sdk 版本低,不要设置。

    // app/build.gradle.ktsandroid {// .......buildFeatures {aidl=true}
    }
    

    配置完成并 sync 之后,会在 app 模块的 main 目录中生成 aidl 目录。

    src/
    ├── main
    │   ├── aidl
    │   │   └── com
    │   │       └── sanren1024
    │   │           └── aidlserver
    
  2. 在目录中创建 .aidl 文件,并使用 java 语法写接口定义。

    // IServerHandle.aidl
    package com.sanren1024.aidlserver;interface IServerHandle {int getPid();
    }
    

    在 IDE 中进行一次编译,可以 build/ 目录中查看到生成的 .java 文件。

  3. 实现 Service ,重写 onBind() 方法,将服务端 Binder 对象公开给调用端。

    在服务端定义一个 Service 类,并定义实现 IServerHandle.Stub 的本地实现类,实现接口中定义的方法。

    public class ServerService extends Service {@Overridepublic IBinder onBind(Intent intent) {Log.i("AIDLServer", "service launch");return LocalServiceBinder.getService();}public static class LocalServiceBinder extends IServerHandle.Stub {public static LocalServiceBinder getService() {return new LocalServiceBinder();}@Overridepublic int getPid() throws RemoteException {return Process.myPid();}}
    }
    

客户端

在客户端要调用服务端的接口,也需要同样包名的的 .aidl 接口定义。

  1. 在客户端的 src/ 目录下使用与服务端同样的 .aidl 文件。

    在这里插入图片描述

  2. 在需要调用远程方法的 Activity 内使用 bindService() 方式进行定义。

    // XxxActivity.javaprivate IServerHandle mServerHandle;
    private final ServiceConnection mSC = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {mServerHandle = IServerHandle.Stub.asInterface(service);}@Overridepublic void onServiceDisconnected(ComponentName name) {mServerHandle = null;}
    };
    
  3. 配置调用。从Android 11 开始,客户端 app 调用服务端 app 内定义的 Service,需要在客户端 Manifest 中配置 queries 标签。

    <!-- 声明app需要访问的其他app的组件信息 -->
    <queries><package android:name="com.sanren1024.aidlserver" /><intent><action android:name="com.sanren1024.server.AIDL_CALL" /><category android:name="android.intent.category.DEFAULT" /></intent>
    </queries>
    

AIDL 编译后生成 .java 文件

aidl 接口定义完成后,由 aidl 工具将 .aidl 文件转成 .java 文件。下图是只包含一个接口定义的 .aidl 文件编译后生成的 java 文件。包含外层接口定义,内部生成两个静态类,及方法定义。
在这里插入图片描述

具体展开生成内容如下。

package com.sanren1024.aidlserver;// 生成了接口类,IServerHandle 继承了 android.os.IInterface。内部有两个静态类都实现 IServerHandle 接口。
public interface IServerHandle extends android.os.IInterface
{// 服务端 IServerHandle 的默认实现。public static class Default implements com.sanren1024.aidlserver.IServerHandle{@Override public int getPid() throws android.os.RemoteException{return 0;}@Overridepublic android.os.IBinder asBinder() {return null;}}// 客户端使用 IServerHandle 的实现。public static abstract class Stub extends android.os.Binder implements com.sanren1024.aidlserver.IServerHandle{private static final java.lang.String DESCRIPTOR = "com.sanren1024.aidlserver.IServerHandle";/** Construct the stub at attach it to the interface. */public Stub(){this.attachInterface(this, DESCRIPTOR);}public static com.sanren1024.aidlserver.IServerHandle asInterface(android.os.IBinder obj){if ((obj==null)) {return null;}android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);if (((iin!=null)&&(iin instanceof com.sanren1024.aidlserver.IServerHandle))) {return ((com.sanren1024.aidlserver.IServerHandle)iin);}return new com.sanren1024.aidlserver.IServerHandle.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{java.lang.String descriptor = DESCRIPTOR;switch (code){case INTERFACE_TRANSACTION:{reply.writeString(descriptor);return true;}case TRANSACTION_getPid:{data.enforceInterface(descriptor);int _result = this.getPid();reply.writeNoException();reply.writeInt(_result);return true;}default:{return super.onTransact(code, data, reply, flags);}}}private static class Proxy implements com.sanren1024.aidlserver.IServerHandle{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 int getPid() throws android.os.RemoteException{android.os.Parcel _data = android.os.Parcel.obtain();android.os.Parcel _reply = android.os.Parcel.obtain();int _result;try {_data.writeInterfaceToken(DESCRIPTOR);boolean _status = mRemote.transact(Stub.TRANSACTION_getPid, _data, _reply, 0);if (!_status && getDefaultImpl() != null) {return getDefaultImpl().getPid();}_reply.readException();_result = _reply.readInt();}finally {_reply.recycle();_data.recycle();}return _result;}public static com.sanren1024.aidlserver.IServerHandle sDefaultImpl;}static final int TRANSACTION_getPid = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);public static boolean setDefaultImpl(com.sanren1024.aidlserver.IServerHandle impl) {if (Stub.Proxy.sDefaultImpl != null) {throw new IllegalStateException("setDefaultImpl() called twice");}if (impl != null) {Stub.Proxy.sDefaultImpl = impl;return true;}return false;}public static com.sanren1024.aidlserver.IServerHandle getDefaultImpl() {return Stub.Proxy.sDefaultImpl;}}public int getPid() throws android.os.RemoteException;
}

其中类结构图如下。

在这里插入图片描述

查看方法

在生成的代码中会看到 attachInterface(IInterface, String) queryLocalInterface(String) 两个方法,都定义在 Binder 类中。

// Binder.javaprivate IInterface mOwner;
@Nullable
private String mDescriptor;// 设置 Binder 成员变量。
public void attachInterface(@Nullable IInterface owner, @Nullable String descriptor) {mOwner = owner;mDescriptor = descriptor;
}// 返回 attachInterface 设置进来的 IInterface 类型变量。
public @Nullable IInterface queryLocalInterface(@NonNull String descriptor) {if (mDescriptor != null && mDescriptor.equals(descriptor)) {return mOwner;}return null;
}

attachInterface

attachInterface 函数的实现很简单,只是设置了 Binder 的成员变量 mOwner mDescription ,用以在查询时使用。

queryLocalInterface

返回 mOwner 对象,这个从 attachInterface 设置入。

分析

在 app 进程间调用,需要 Binder 驱动调用。

在服务端,Binder 的本地代理实现定义在一个Service 类内,并通过 onBinder 方法将 Binder 对象公开给调用端。在客户端需要调用服务端方法的时候,先通过 bindService() 方法连接远程 Service,获取远程接口,再调用相应方法(调用 YourInterfaceName.Stub.asInterface((IBinder) service),以将返回的参数转换为 YourInterface 类型。)。Service 创建时,会分别调用到 Service.onCreate() Service.onBind(Intent) 方法。再从客户端方面,通过 onServiceConnected(ComponentName, IBinder) 将收到一个 IBinder 实例(名为 service)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/175252.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

虚幻C++基础 day1

虚幻C概念 虚幻C类的继承结构 虚幻引擎C类层级结构(Hierarchy) 这些基本类又派生出了很多子类&#xff0c;例&#xff1a; UE中的反射与垃圾回收系统 例如一个创建了一个Actor类&#xff0c;有一个Actor类型指针去指向这个Actor类&#xff0c;如果的指针被销毁了&#xff…

38基于matlab的期货预测,利用PSO优化SVM和未优化的SVM进行对比,得到实际输出和期望输出结果。

基于matlab的期货预测&#xff0c;利用PSO优化SVM和未优化的SVM进行对比&#xff0c;得到实际输出和期望输出结果。线性核函数、多项式、RBF核函数三种核函数任意可选&#xff0c;并给出均方根误差&#xff0c;相对误差等结果&#xff0c;程序已调通&#xff0c;可直接运行。 3…

谈API接入必须了解的各大API调用电商API应用场景

哪些业务场景可以使用API接口&#xff1f; &#xff08;1&#xff09;爬虫业务&#xff1a;在爬虫业务中&#xff0c;使用API接口可以帮助解决IP限制、反爬虫策略等问题&#xff0c;提高爬取数据的效率和稳定性。 &#xff08;2&#xff09;网络安全&#xff1a;在网络安全领…

虚拟化、容器与Docker基本介绍以及安装部署(Docker 基本管理)

虚拟化、容器与Docker基本介绍以及安装部署&#xff08;Docker 基本管理&#xff09; 1、Docker 概述1.1Docker与虚拟机的区别1.2容器在内核中支持2种重要技术&#xff1a;1.3Docker核心概念 2、安装docker服务docker安装步骤详解 3、 网络优化4、docker基本命令4.1查看镜像——…

代码随想录算法训练营第三十九天丨 动态规划part02

62.不同路径 思路 动态规划 机器人从(0 , 0) 位置出发&#xff0c;到(m - 1, n - 1)终点。 按照动规五部曲来分析&#xff1a; 确定dp数组&#xff08;dp table&#xff09;以及下标的含义 dp[i][j] &#xff1a;表示从&#xff08;0 &#xff0c;0&#xff09;出发&#…

荣电集团与钕希科技签署全面战略合作

10月26日&#xff0c;荣电集团&#xff08;以下简称荣电&#xff09;与钕希科技南京有限公司&#xff08;以下简称钕希科技&#xff09;今天在合肥市签署全面战略合作协议&#xff0c;联合进军混合现实&#xff08;Mixed Reality&#xff0c;以下简称MR&#xff09;空间计算高科…

leetcode-字符串

1.反转字符串LeetCode344. 20230911 难度为0&#xff0c;此处就不放代码了 注意reverse和swap等一系列字符串函数什么时候该用&#xff0c;记一记库函数 swap可以有两种实现&#xff0c;涨知识了&#xff0c;除了temp存值还可以通过位运算&#xff1a;s[i] ^ s[j]; s[j] ^ s[i…

【c++|opencv】二、灰度变换和空间滤波---1.灰度变换、对数变换、伽马变换

every blog every motto: You can do more than you think. https://blog.csdn.net/weixin_39190382?typeblog 0. 前言 灰度变换、对数变换、伽马变换 1. 灰度变换 #include <iostream> #include <opencv2/opencv.hpp>using namespace std; using namespace c…

03_Flutter自定义下拉菜单

03_Flutter自定义下拉菜单 在Flutter的内置api中&#xff0c;可以使用showMenu实现类似下拉菜单的效果&#xff0c;或者使用PopupMenuButton组件&#xff0c;PopupMenuButton内部也是使用了showMenu这个api&#xff0c;但是使用showMenu时&#xff0c;下拉面板的显示已经被约定…

Redis(10)| I/O多路复用(mutiplexing)

上文select/epoll 在上文《Redis&#xff08;09&#xff09;| Reactor模式》 思考问题可以使用I/O多路复用技术解决多多客户端TCP连接问题&#xff0c;同时也提到为了解决最早期的UNIX系统select调用存在的四个问题。 select(int nfds, fd_set *r, fd_set *w, fd_set *e, stru…

动手学深度学习——第七次学

LeNet&#xff08;LeNet-5&#xff09;由两个部分组成&#xff1a; 卷积编码器和全连接层密集块 卷积把高宽不断变小&#xff0c;把通道数逐渐增多&#xff0c;&#xff08;最后高宽会变成&#xff0c;通道会变得很大&#xff0c;然后做全连接进行输出&#xff09;通道信息可以…

7+共病思路。WGCNA+多机器学习+实验简单验证,易操作

今天给同学们分享一篇共病WGCNA多机器学习实验的生信文章“Shared diagnostic genes and potential mechanism between PCOS and recurrent implantation failure revealed by integrated transcriptomic analysis and machine learning”&#xff0c;这篇文章于2023年5月16日发…

人到中年应该怎么交朋友

听人劝、吃饱饭,奉劝各位小伙伴,不要订阅该文所属专栏。 作者:不渴望力量的哈士奇(哈哥),十余年工作经验, 跨域学习者,从事过全栈研发、产品经理等工作,现任研发部门 CTO 。荣誉:2022年度博客之星Top4、博客专家认证、全栈领域优质创作者、新星计划导师,“星荐官共赢计…

虚拟机克隆

linux系统的组成&#xff1b; 主根目录和根目录; 所有的根目录都包含在主根目录中&#xff1b; 根目录&#xff1a; /root /home/xxx,yyy,zzz;主根目录&#xff1b;/ 一个重要的子目录&#xff1a;etc passwd, 保存了所有的三类用户信息&#xff1b;bashrc, 可以设置别名 及…

HarmonyOS开发:基于http开源一个网络请求库

前言 网络封装的目的&#xff0c;在于简洁&#xff0c;使用起来更加的方便&#xff0c;也易于我们进行相关动作的设置&#xff0c;如果&#xff0c;我们不封装&#xff0c;那么每次请求&#xff0c;就会重复大量的代码逻辑&#xff0c;如下代码&#xff0c;是官方给出的案例&am…

阿里云推出通义千问App,提供全方位的协助

&#x1f989; AI新闻 &#x1f680; 阿里云推出通义千问App&#xff0c;提供全方位的协助 摘要&#xff1a;阿里云旗下大模型通义千问App登陆各大安卓应用市场&#xff0c;具有超大规模预训练模型&#xff0c;可在创意文案、办公助理、学习助手、趣味生活等方面协助用户。功…

Transformer.js简明教程【Web AI】

Transformers.js 可在你的 Web 浏览器中实现最先进的机器学习&#xff0c;无需服务器。 它提供预训练模型和熟悉的 API&#xff0c;支持自然语言处理、计算机视觉、音频和多模态领域的任务。 借助 Transformers.js&#xff0c;开发人员可以直接在浏览器中运行文本分类、图像分类…

RPC与HTTP的关系

首选理清楚关系 RPC与HTTP是两个不同维度的东西 HTTP 协议&#xff08;Hyper Text Transfer Protocol&#xff09;&#xff0c;又叫做超文本传输协议&#xff0c;是一种传输协议&#xff0c;平时通过浏览器浏览网页网页&#xff0c;用到的就是 HTTP 协议。 而 RPC&#xff0…

Flutter FittedBox

&#x1f525; 英文单词FittedBox &#x1f525; Fitted 通过有道翻译如下 &#xff1a; Box 通过有道翻译如下 &#xff1a; 对 FittedBox 的理解 我们可以将 FittedBox 理解为合适的盒子&#xff0c;将其它布局放到FittedBox这样一个盒子中&#xff0c;从而实现 盒子里面的…

ceph高可用

配置基础环境 # 关闭防火墙 systemctl stop firewalld systemctl disable firewalld# 关闭selinux setenforce 0 sed -i s/^SELINUX.*/SELINUXdisabled/ /etc/selinux/config 安装基础环境 然后安装ceph的密钥&#xff0c;centos7和8都要执行&#xff0c;下面不特别说明都是c…