前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >Android进程间通信 -- Binder学习记录

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

作者头像
用户6414482
发布于 2022-08-28 00:57:34
发布于 2022-08-28 00:57:34
53400
代码可运行
举报
文章被收录于专栏:技闲说技闲说
运行总次数:0
代码可运行

前言


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

  • What
  • Why
  • How

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的工作原理:

可以看出无论是服务的注册还是服务的获取,都需要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上,因此手机安全性显得尤为重要。

而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通信架构图:

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

从调用过程上看可以发现,当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端的线程。

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

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
运行
AI代码解释
复制
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系统。如果想全部写明白要写好几篇文章。后续应该会出专门的几篇文章从源码入手,详细的阐述Android Binder。

作者的话

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

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

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

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-07-22,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 技闲说 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Android进程间通信之一:Binder机制学习
主要是驱动设备的初始化(binder_init),打开 (binder_open),映射(binder_mmap),数据操作(binder_ioctl)。
北洋
2021/12/08
5210
Android进程间通信之一:Binder机制学习
android进程间通信之Binder学习笔记(一)
Binder 是一种进程间通信机制,基于开源的 OpenBinder 实现;OpenBinder 起初由 Be Inc. 开发,后由 Plam Inc. 接手。
李小白是一只喵
2020/04/24
8050
Android-Binder机制
Binder机制是​ Android系统中进程间通讯(IPC)的一种方式,Android中ContentProvider、Intent、aidl都是基于Binder。
用户7557625
2020/07/15
1.3K0
Android-Binder机制
Binder机制 简单理解
Android系统中,涉及到多进程间的通信底层都是依赖于Binder IPC机制。例如当进程A中的Activity要向进程B中的Service通信,这便需要依赖于Binder IPC。不仅于此,整个Android系统架构中,大量采用了Binder机制作为IPC(进程间通信)方案。
zhangjiqun
2024/12/16
1370
Binder机制 简单理解
Android Binder实现浅析-Binder驱动
Android是如何实现跨进程通信的,大家熟悉的Binder是什么,怎么设计的,进程间的数据如何发送接收的。本文将以及解析,并对Binder驱动实现、Native层实现、Java层实现三块做一个总结分析。
233333
2020/02/18
9540
Android Binder实现浅析-Binder驱动
Android Binder跨进程通信
为了保证 安全性 & 独立性,一个进程 不能直接操作或者访问另一个进程,即Android的进程是相互独立、隔离的
Anymarvel
2018/10/22
8950
Android Binder跨进程通信
写给Android开发的Binder指南
Linux 已经提供了管道、消息队列、共享内存和 Socket 等 IPC 机制。那为什么 Android 还要提供 Binder 来实现 IPC 呢?主要是基于性能、稳定性和安全性几方面的原因。
老马的编程之旅
2022/06/22
5340
写给Android开发的Binder指南
写给 Android 应用工程师的 Binder 原理剖析
这篇文章我酝酿了很久,参考了很多资料,读了很多源码,却依旧不敢下笔。生怕自己理解上还有偏差,对大家造成误解,贻笑大方。又怕自己理解不够透彻,无法用清晰直白的文字准确的表达出 Binder 的设计精髓。直到今天提笔写作时还依旧战战兢兢。
张磊BARON
2018/04/13
1.9K0
写给 Android 应用工程师的 Binder 原理剖析
Android 进阶8:进程通信之 Binder 机制浅析
张拭心 shixinzhang
2018/01/05
2.2K0
Android 进阶8:进程通信之 Binder 机制浅析
Android Framework学习(九)之Binder概述
Android系统中,每个应用程序是由Android的Activity,Service,Broadcast,ContentProvider这四大组件的中一个或多个组合而成,这四大组件所涉及的多进程间的通信底层都是依赖于Binder IPC机制。不仅于此,整个Android系统架构中,大量采用了Binder机制作为IPC(进程间通信)方案,当然也存在部分其他的IPC方式,比如Zygote通信便是采用socket。
老马的编程之旅
2022/06/22
4250
Android Framework学习(九)之Binder概述
Binder纯理论分析
首先Binder是Android中的一种独有的跨进程通信方式,简称IPC。它是专门为Android平台设计的。
Rouse
2021/01/24
7770
Android跨进程通信IPC之6——Binder框架
为了让大家更好的理解Binder机制,我们先来看下Android的整体架构。因为这样大家就知道在Android架构中Binder出于什么地位。 用一下官网上的图片
隔壁老李头
2018/08/30
1.4K0
Android跨进程通信IPC之6——Binder框架
Android Binder面试详解
目前linux支持的IPC包括传统的管道、System V IPC、即消息队列/共享内存/信号量,以及socket中只有socket支持Client-Server的通信方式
用户2802329
2018/08/07
1K0
Android Binder面试详解
深入理解Binder
之前一直对 Binder 理解不够透彻,仅仅知道一些皮毛,所以最近抽空深入理解一下,并在这里做个小结。
俞其荣
2019/07/09
1.2K0
深入理解Android IPC机制之Binder机制
Binder是Android系统进程间通信(IPC)方式之一。Linux已经拥有的进程间通信IPC手段包括(Internet Process Connection): 管道(Pipe)、信号(Signal)和跟踪(Trace)、插口(Socket)、报文队列(Message)、共享内存(Share Memory)和信号量(Semaphore)。本文详细介绍Binder作为Android主要IPC方式的优势。 Binder机制概述: 基于Client-Server的通信方式广泛应用于从互联网和数据库访问到嵌
xiangzhihong
2018/02/05
1.1K0
深入理解Android IPC机制之Binder机制
面试 | 再也不怕被问 Binder 机制了
原文链接:https://juejin.cn/post/7293175592162836514
GeeJoe
2023/10/24
1.4K1
面试 | 再也不怕被问 Binder 机制了
Android高频面试专题 - 进阶篇(三)Binder机制
Android作为移动端操作系统,传统的Linux进程间通信机制不满足于Android,所以开发了一套新的IPC机制,就是Binder机制。
Android扫地僧
2020/03/19
9800
Android高频面试专题 - 进阶篇(三)Binder机制
图解Android中的binder机制
Binder做为Android中核心机制,对于理解Android系统是必不可少的,关于binder的文章也有很多,但是每次看总感觉看的不是很懂,到底什么才是binder机制?为什么要使用binder机制?binder机制又是怎样运行的呢?这些问题只是了解binder机制是不够的,需要从Android的整体系统出发来分析,在我找了很多资料后,真正的弄懂了binder机制,相信看完这篇文章大家也可以弄懂binder机制。
做个快乐的码农
2021/11/26
9880
图解Android中的binder机制
一文图解Android的Binder机制
Binder做为Android中核心机制,对于理解Android系统是必不可少的,关于binder的文章也有很多,但是每次看总感觉看的不是很懂,到底什么才是binder机制?为什么要使用binder机制?binder机制又是怎样运行的呢?这些问题只是了解binder机制是不够的,需要从Android的整体系统出发来分析,在我找了很多资料后,真正的弄懂了binder机制,相信看完这篇文章大家也可以弄懂binder机制。
没关系再继续努力
2021/11/25
5740
Linux和Android的IPC通信简介
IPC全名为inter-Process Communication,含义为进程间通信,是指两个进程之间进行数据交换的过程。在Android和Linux中都有各自的IPC机制,这里分别来介绍下。
提莫队长
2020/06/02
1.8K0
推荐阅读
相关推荐
Android进程间通信之一:Binder机制学习
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验