Android进程间通信 — Binder学习记录

前言

移动设备的操作系统阵营之一的Android,底层基于Linux内核,中间为Native&Runtime层和Framework层。我们知道Linux本身有着很成熟的IPC(进程间通信)机制,比如管道、消息队列、共享内存、socket、信号和信号量等。然而,Android却使用Binder来作为它的IPC的方案,这是为何呢?接下来,就把我之前学习Binder的心得写下来。

WhatWhyHow

What — Binder是什么

Binder起源于OpenBinder。

官方正式的说,Binder是Android跨进程通信的方式,采用了C/S架构。主要包括4个组件,Client、Server、ServiceManager和Binder驱动。其中Client、Server和ServiceManager运行在用户空间,Binder驱动运行在内核空间。

说完了没说一样是吧。说白了其实Binder对于Android来说就像是电话对于人一样,用于通信的。其中Client、Server、ServiceManager和Binder驱动就是电话内部的几个组成组件。

Client:进程间通信的客户端

Server:进程间通信的服务端

ServiceManager:Binder服务的大管家,它是一个守护进程,Client端和Server端的相互通信都需要借助于它。

下面这张图就是Binder的工作原理:

Android进程间通信 -- Binder学习记录

可以看出无论是服务的注册还是服务的获取,都需要ServiceManager。

Why — 为何使用Binder

刚刚提到说,Android底层基于Linux内核,Linux已经包含了好多成熟的进程间通信的方案,为何要选用Binder呢?

这里从我总结的几点说。

1. 性能:Binder在一次通信中只进行一次内存拷贝。而管道、消息队列、socket等都需要2次内存拷贝,可别小看这一倍带来的差距。对于移动设备来说,性能一直是个大问题。想象一下,如果选用管道作为IPC的方案的话,那么Android绘制屏幕的时候都需要与WindowManager进行IPC通信,如果这里效率不高,势必会造成卡顿甚至于app卡死。

当然了,共享内存的方式压根就不需要内存拷贝,但是共享内存在实际写代码会非常复杂,需要频繁的加锁释放锁,稍有不慎就会造成死锁。

所以从性能的角度来说,Binder的性能仅次于共享内存。

IPC方式

内存拷贝次数

socket/管道/消息队列

2

共享内存

0

Binder

1

2. 安全性:Android作为开源的移动端操作系统,拥有非常多的开发平台,有用于手机设备,有用于智能穿戴设备,也有嵌入式及其机器人平台。同时运行的app来源也很多,很多不正规的第三方app都可以发布、安装到Android上,因此手机安全性显得尤为重要。

通义听悟 通义听悟

阿里云通义听悟是聚焦音视频内容的工作学习AI助手,依托大模型,帮助用户记录、整理和分析音视频内容,体验用大模型做音视频笔记、整理会议记录。

通义听悟 85 查看详情 通义听悟

而Linux传统的IPC的方案,接收方都无法获取对方进程的身份,无法判断对方是否可靠,Linux传统的IPC方案没有任何保护措施。

在Android系统中,Android为每一个安装的app都分配了一个UID标识,这个UID就作为鉴别对方进程身份的重要标志。而Binder采用的C/S架构,系统只暴露了Client端,不会暴露Server端。实际通信中Server端会判断对方进程的UID是否满足权限(在Android 6.0以上的系统,都是通过弹窗询问)。传统的IPC方案只能在用户空间的数据包里填充UID,只能在用户空间定义协议。Binder可以在Binder驱动程序中进行分配和填充记录UID的操作,也就是说,Binder机制对于通信双方的身份是在内核中进行校验支持的,安全性更高。

3. 灵活性:Binder采用的C/S架构,Client端和Server端实则都借助于ServiceManager,在我看来ServiceManager相当于控制和服务中心,有着统一中心化管理的能力,在ServiceManager注册服务后,就可以统一的发布Service供其他进程使用。所以它更加的灵活。

4. 稳定性:相较于Linux传统IPC方案,共享内存虽然性能更好,但是操作起来复杂,需要控制好锁以及各种同步的场景,稍有不慎就会死锁;而B/S架构的模式,像管道/消息队列还得进行包装。而Binder采用C/S架构,稳定性更好。

通过上述的4点,Binder就很适合作为Android系统的IPC方案了。

How — Binder的架构

这里先放出Binder通信架构图:

Android进程间通信 -- Binder学习记录

一次Binder通信过程会涉及到app层、Framework层、Native层和Kernel层,算是把Android的架构从上到下走了个遍。假设已经创建了自己的Client、Service类,并且在Client端持有Service的引用,并且申请调用Service的A函数,那么在Android系统内部的调用过程是这样的:

Android进程间通信 -- Binder学习记录

从调用过程上看可以发现,当Client端发起请求后,Client端的当前线程会挂起。这里需要注意,如果Server端进程需要执行长时间的操作的话,最好不要在UI线程里做以免发生ANR。

接下来说说Binder的C/S架构。我们知道Android系统在开机启动过程中Zygote创建后会fork出system_server进程,然后由system_server进程会孵化出大量的系统服务,分为引导服务、核心服务和其他服务(这里后续会出一篇文章具体介绍)。那么在Android系统中是怎么管理这些服务的,并且让用户可以跨进程调用这些服务呢?

调用系统服务的过程:Android在开机过程初始化所有的Service后,会将这些Service向ServiceManager注册,Client端想要申请具体的Service将直接向ServiceManager要就行了。具体做法是Client端首先向ServiceManager查询,得到具体的Service的引用,然后通过这个引用向具体的Server端发送请求,Server端执行完成后就返回。

而这里有个问题,Client端持有Service的引用,然后调用具体Server端的某个函数,一次Binder请求就完成了。可是Client端和Server端的这两个进程之间是怎么共享资源的呢?聪明的读者一定都猜到了,没错,就是通过Binder驱动。

当Client向Server发请求时,Client会先从自己的进程空间把请求数据拷贝到内核空间,然后因为Server和内核共享数据,所以这里不需要再进行拷贝,直接通过内存地址偏移量获得请求数据地址,并通知Server端执行onTransact()函数,这个函数属于Binder类,当Server端进程执行完毕后将结果写入自己的共享内存中,Binder驱动再将结果拷贝到Client端的进程空间,并唤醒Client端的线程。

Android进程间通信 -- Binder学习记录

接下来说说Binder驱动。用户态程序通过系统调用陷入内核态,比如打开Binder驱动方法的调用链为:open() –> __open() –> binder_open()。open()为用户态方法,__open()为系统调用,通过系统调用表查找到对应内核Binder驱动的binder_open()方法。

Android进程间通信 -- Binder学习记录

Binder驱动有几个核心的方法:

binder_init:主要工作是注册misc设备。binder_open:打开Binder驱动设备。期间会创建binder_proc对象(管理着IPC所需的各种信息)。binder_mmap:在内核虚拟地址空间申请一块与用户虚拟内存相同大小的内存;然后再申请1个page大小的物理内存,再将同一块物理内存分别映射到内核虚拟地址空间和用户虚拟内存空间,从而实现了用户空间的Buffer和内核空间的Buffer同步操作的功能。刚刚讲C/S架构时提到Server端不需要拷贝,Server和内核空间共享一块空间主要就是通过binder_mmap()来实现的。binder_ioctl:负责在两个进程间收发IPC数据和IPC reply数据。

具体的源码分析这里就不展开了。

Binder机制运用。首先看一下下面2行代码:

代码语言:javascript代码运行次数:0运行复制

TelephonyManager telephonyManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);  String deviceId = telephonyManager.getDeviceId();

getSystemService(Context.TELEPHONY_SERVICE)在Android系统内部会向ServiceManager查询标识为TELEPHONY_SERVICE的Server端对象的引用。即TelephonyManager对象的引用,这个引用的真正实现是TelephonyManager的某个代理。得到这个引用后,调用getDeviceId时,真正的实现是在代理里面,代理把参数、数据等打包到Parcel对象中,然后调用transact函数(该函数继承于Binder),之后就会触发Binder驱动做一些列操作,Client端把数据拷贝到内核空间,然后Server端接收,执行完相应的操作(这里是getDeviceId)后把结果映射到内核空间,然后内核再拷贝到Client端的进程空间中从而完成这次Binder IPC调用。

Android进程间通信 -- Binder学习记录

总结

这篇文章记录了个人学习Android Binder的见解,算是一个概要性的学习记录和总结。因为毕竟写的比较简单,也还没有从Android源码来分析,因为Binder贯穿着整个Android系统。如果想全部写明白要写好几篇文章。后续应该会出专门的几篇文章从源码入手,详细的阐述Android Binder。

作者的话

个人喜欢计算机技术,主要涉及的领域包括:Android系统,Linux内核,嵌入式软/硬件,机器人和智能硬件。同时也对其他的各个技术栈都感兴趣。

同时也很喜欢生活,喜欢享受生活,喜欢用拍照和视频的方式来记录生活。

如果你也是个爱学习爱技术的人,欢迎一起探讨,没准,咱们能成为好朋友。如果觉得本文有哪些不对的地方,欢迎指出,大家一起学习进步。

以上就是Android进程间通信 — Binder学习记录的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/300202.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
绘蛙AI修图怎样制作高点击率主图?电商运营技巧
上一篇 2025年11月5日 00:31:25
夸克AI官方网站主页链接 夸克AI智能助手平台入口官方地址
下一篇 2025年11月5日 00:31:29

相关推荐

  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • c++如何实现UDP通信_c++基于UDP的网络通信示例

    UDP通信基于套接字实现,适用于实时性要求高的场景。1. 流程包括创建套接字、绑定地址(接收方)、发送(sendto)与接收(recvfrom)数据、关闭套接字;2. 服务端监听指定端口,接收客户端消息并回传;3. 客户端发送消息至服务端并接收响应;4. 跨平台需处理Winsock初始化与库链接,编…

    2026年5月10日
    000
  • 谷歌浏览器如何截图 谷歌浏览器页面截图技巧

    谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧

    使用谷歌浏览器的开发者工具截图步骤:1. 按ctrl+shift+i(windows/linux)或cmd+option+i(mac)打开开发者工具。2. 点击右上角三个点,选择”更多工具”,再选择”截图”。3. 选择截取整个页面。推荐的谷歌浏览器扩展…

    2026年5月10日 用户投稿
    100
  • Golang使用Protobuf定义接口与消息格式

    Protobuf通过字段编号实现兼容性,新增字段可忽略、删除字段可保留编号,确保新旧版本互操作,支持服务独立演进。 在Golang项目中,利用Protobuf定义接口和消息格式,本质上是为服务间通信构建了一套高效、类型安全且跨语言的契约。它让数据结构清晰可见,RPC调用标准化,极大地简化了分布式系统…

    2026年5月10日
    000
  • pycharm解析器怎么添加 解析器添加详细流程

    在pycharm中添加解析器的步骤包括:1) 打开pycharm并进入设置,2) 选择project interpreter,3) 点击齿轮图标并选择add,4) 选择解析器类型并配置路径,5) 点击ok完成添加。添加解析器后,选择合适的类型和版本,配置环境变量,并利用解析器的功能提高开发效率。 在…

    2026年5月10日
    000
  • 硬盘数据被误删除怎么办?教你快速找回删除的文件!

    硬盘数据被误删除,别慌!恢复数据并非不可能,关键在于你接下来的操作。立刻停止对该硬盘的任何写入操作,然后尝试使用专业的数据恢复软件。 解决方案 首先,数据恢复的原理是,删除文件后,操作系统只是将文件占用的空间标记为“可覆盖”,但文件本身的数据可能还存在于硬盘上。所以,避免新的数据写入覆盖掉旧数据,是…

    2026年5月10日
    000
  • Python官网用户调查的参与方式_Python官网反馈提交详细教程

    答案是通过访问Python官网新闻页面、邮件邀请链接或GitHub仓库提交反馈。具体为:访问官网查找用户调查公告,或点击邮件中的专属链接参与,在GitHub的cpython仓库提交技术建议,并注意如实填写问卷与保护隐私。 如果您希望参与Python官网的用户调查并提交反馈,可以通过官方指定的渠道完成…

    2026年5月10日
    000
  • JavaScript Electron桌面应用

    答案:使用JavaScript开发%ignore_a_1%桌面应用需结合Web技术与Node.js,通过主进程管理窗口、渲染进程展示界面,并利用IPC通信,调用系统功能如文件对话框,最后用electron-builder打包发布,注意安全与进程职责分离。 用JavaScript开发Electron桌…

    2026年5月10日
    000
  • Go语言连接外部MySQL数据库:DSN配置与常见错误解析

    本文详细阐述了go语言使用`go-sql-driver/mysql`驱动连接外部mysql数据库的正确方法。重点介绍了数据源名称(dsn)的规范格式,特别是主机地址部分的配置,以避免常见的“getaddrinfow: the specified class was not found.”等网络解析错…

    2026年5月10日
    000
  • Linux文件系统iostat命令使用技巧

    Linux文件系统iostat命令使用技巧Linux文件系统iostat命令使用技巧Linux文件系统iostat命令使用技巧Linux文件系统iostat命令使用技巧

    iostat是Linux系统中用于监控I/O设备负载的关键工具,能分析磁盘性能并识别瓶颈。默认输出包括CPU使用率和设备I/O统计,分为系统启动以来的平均值和当前采样周期数据。核心指标有:%util反映设备利用率,持续接近100%可能表示I/O瓶颈;await为平均I/O等待时间,过高说明响应变慢;…

    2026年5月10日 用户投稿
    000
  • 如何测试html5编码_测试HTML5页面编码兼容性方法【编码测试】

    HTML5页面编码兼容性测试需五步:一查meta charset是否正确且前置;二验HTTP响应头Content-Type charset是否为utf-8;三用file或chardet工具探测实际编码;四跨浏览器测试URL参数中中文、Emoji解析;五通过W3C验证服务检查编码声明与字节一致性。 如…

    2026年5月10日
    100
  • 后缀php怎么打开_php文件打开方式与运行环境搭建指南

    要打开PHP文件需根据用途选择方式:查看代码可用文本编辑器或IDE,运行则需服务器环境。推荐新手使用XAMPP、WAMP等集成环境,将文件放入htdocs目录后访问localhost;开发者可利用PHP内置服务器,命令行执行php -S localhost:8000运行;高级用户可手动配置Apach…

    2026年5月10日
    000
  • 解决Python脚本中相对路径文件找不到的常见问题与策略

    本文旨在解决python脚本中因相对路径处理不当导致的文件找不到错误,尤其是在项目迁移后。文章将深入探讨python中相对路径的工作原理、当前工作目录(cwd)的影响,并提供使用`os.getcwd()`诊断问题以及利用`os.path.dirname(__file__)`结合`os.path.jo…

    2026年5月10日
    000
  • Golang如何提升TCP长连接处理效率_Golang TCP长连接处理性能优化实践详解

    答案:通过非阻塞I/O、单Goroutine双工模型、sync.Pool对象复用、TCP_NODELAY优化及高效心跳管理,结合系统调优,可显著提升Golang百万级TCP长连接处理效率。 在高并发网络服务场景中,TCP长连接的处理效率直接影响系统的吞吐能力和资源消耗。Golang凭借其轻量级Gor…

    2026年5月10日
    000
  • C++内存检测工具 Valgrind使用实践指南

    Valgrind是一款主要用于Linux和macOS的内存调试工具,可检测内存泄漏、越界访问、未初始化内存使用等问题,通过memcheck工具结合–leak-check=full、–track-origins=yes等选项进行详细分析,需编译时添加-g选项以支持调试信息,虽然…

    2026年5月10日
    000
  • php源码怎么运行手机_php源码手机运行环境搭建步骤【教程】

    可在手机上通过特定工具运行PHP源码。首先选择支持PHP的移动应用,安卓用户可安装UserLAnd或KSWEB,iOS用户可尝试iSH Shell或a-Shell;然后配置本地服务器环境,启动HTTP和PHP服务,将PHP文件放入指定根目录;接着可通过Termux搭建完整开发环境,更新包列表并安装P…

    2026年5月10日
    200
  • Go语言:检查预编译库的构建版本与平台信息

    本文详细介绍了如何利用go语言内置的`go tool pack`工具,从预编译的go静态库(`.a`文件)中提取其构建信息,包括go编译器版本、操作系统和cpu架构。当`go build`因库版本不匹配而失败时,此方法能帮助开发者准确诊断问题,确保构建环境与库的兼容性。 在Go语言的开发实践中,我们…

    2026年5月10日
    000
  • PHP动态网页数据库备份恢复_PHP动态网页MySQL数据库备份教程

    答案:PHP动态网页的MySQL数据库备份与恢复需通过定期导出SQL文件并安全存储来保障数据安全,核心方法包括使用mysqldump命令行工具实现高效灵活的自动化备份,利用phpMyAdmin图形化工具进行手动导出导入以降低操作门槛,以及通过PHP脚本调用系统命令将备份过程集成到应用中;恢复时可采用…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信