当前位置: 首页 > news >正文

[Android] Binder all-in-all

前言:

Binder 是一种 IPC 机制,使用共享内存实现进程间通讯,既可以传递消息,也可以传递创建在共享内存中的对象,而Binder本身就是用共享内存实现的,因此遵循Binder写法的类是可以实例化后在进程间传递的。

Binder在Android架构中有很重的地位,各个模块都在重度使用它,从代码可读性角度看,在为熟悉之前的可读性较差。从整体架构的角度看,在各个模块中通过IPC分离接口和实现有效地提高了扩展性。

写法:

1.定义接口

这个接口需要 Client/Bp 和 Server/Bn 都要遵循,Client需要关注有哪些消息类型,即enum值,Server需要关注虚函数,主要任务是根据消息中的enum值判断不同的业务类型,然后调用相应的虚函数。

/** IDemo.h
*/class IDemo : public IInterface {public:enum {ALERT = IBinder::FIRST_CALL_TRANSACTION,PUSH,ADD};// Sends a user-provided value to the servicevirtual void        push(int32_t data)          = 0;// Sends a fixed alert string to the servicevirtual void        alert()                     = 0;// Requests the service to perform an addition and return the resultvirtual int32_t     add(int32_t v1, int32_t v2) = 0;DECLARE_META_INTERFACE(Demo);
};// This implementation macro would normally go in a cpp file
IMPLEMENT_META_INTERFACE(Demo, "com.my.Demo");

2.服务端实现

服务端需要继承 BnInterface<INTERFACE_NAME>,当前例子里需要继承 BnInferface<IDemo> , BnInterface 有一个纯虚函数onTransact需要实现。

/** BnDemo.hpp
*/class BnDemo : public BnInterface<IDemo> {virtual status_t onTransact(uint32_t code, const Parcel& data,Parcel* reply, uint32_t flags = 0);
};status_t BnDemo::onTransact(uint32_t code, const Parcel& data,Parcel* reply, uint32_t flags) {data.checkInterface(this);switch(code) {case ALERT: {alert();return NO_ERROR;} break;case PUSH: {int32_t inData = data.readInt32();push(inData);return NO_ERROR;} break;case ADD: {int32_t inV1 = data.readInt32();int32_t inV2 = data.readInt32();int32_t sum = add(inV1, inV2);reply->writeInt32(sum);return NO_ERROR;} break;default:return BBinder::onTransact(code, data, reply, flags);}
}

如果不希望把业务代码和Binder架构代码混到一起,那么就再定义一个类来实现业务层。

/** Demo.hpp
*/class Demo : public BnDemo {virtual void push(int32_t data) {// Do something with the data the client pushed}virtual void alert() {// Handle the alert}virtual int32_t add(int32_t v1, int32_t v2) {return v1 + v2;}
};

3.客户端实现

客户端需要继承BpInterface<INTERFACE_NAME>,然后就可以通过 BpInterface 的 remote()->transact 来给 服务端发送消息了。那么如何和服务端关联起来呢?这些工作是Binder内部自行关联的,我们只需要在定义Client的时候指定如下接收 IBinder 入参的构造函数即可。

/*
*  BpDemo.hpp
*/class BpDemo : public BpInterface<IDemo> {public:BpDemo(const sp<IBinder>& impl) : BpInterface<IDemo>(impl) { }virtual void push(int32_t push_data) {Parcel data, reply;data.writeInterfaceToken(IDemo::getInterfaceDescriptor());data.writeInt32(push_data);remote()->transact(PUSH, data, &reply);}virtual void alert() {Parcel data, reply;data.writeInterfaceToken(IDemo::getInterfaceDescriptor());remote()->transact(ALERT, data, &reply, IBinder::FLAG_ONEWAY);}virtual int32_t add(int32_t v1, int32_t v2) {Parcel data, reply;data.writeInterfaceToken(IDemo::getInterfaceDescriptor());data.writeInt32(v1);data.writeInt32(v2);remote()->transact(ADD, data, &reply);int32_t res;status_t status = reply.readInt32(&res);return res;}
};

4.如何使用

服务端注册服务给系统的服务管理器。

defaultServiceManager()->addService(String16("com.my.Demo"), new Demo());

客户端在系统的服务管理器中根据 标识字符串 查找相应的服务,把返回的值cast 为 INTERFACE 实例使用即可。

p<IServiceManager> sm = defaultServiceManager();
sp<IBinder> binder = sm->getService(String16("com.my.Demo"));
sp<IDemo> demo = interface_cast<IDemo>(binder);demo->alert();
demo->push(65);
int32_t sum = demo->add(453, 827);

5.代码阅读

服务端如何查找service的标识字符串?

代码中搜索 addService

客户端如何查找连接了哪个Binder服务端?

代码中搜索 getService

记住上面两点,能够快速定位客户端和服务端的关联关系。

参考:

https://android.googlesource.com/platform/frameworks/native/+/jb-mr1-dev/include/binder/IInterface.h

http://www.lryc.cn/news/260279.html

相关文章:

  • 无人零售柜:快捷舒适购物体验
  • Bash script进阶笔记
  • OpenCV图像处理——Python开发中OpenCV视频流的多线程处理方式
  • webGL开发智慧城市流程
  • Django讲课笔记02:Django环境搭建
  • 黑豹程序员-原生JS拖动div到任何地方-自定义布局
  • <软考高项备考>《论文专题 - 7 论文的项目背景之技术架构》
  • 6.3 C++11 原子操作与原子类型
  • 智能优化算法应用:基于狮群算法3D无线传感器网络(WSN)覆盖优化 - 附代码
  • BERT、GPT学习问题个人记录
  • HeartBeat监控Mysql状态
  • 软件开发经常出现的bug原因有哪些
  • 代码随想录27期|Python|Day15|二叉树|层序遍历|对称二叉树|翻转二叉树
  • 鸿蒙开发组件之Web
  • 成绩分析。
  • Excel实现字母+数字拖拉自动递增,步长可更改
  • Java之Stream流
  • vue中element-ui日期选择组件el-date-picker 清空所选时间,会将model绑定的值设置为null 问题 及 限制起止日期范围
  • 使用模方时,三维模型在su中显示不了怎么办?
  • AR-LDM原理及代码分析
  • MySQL常见死锁的发生场景以及如何解决
  • Leetcode 47 全排列 II
  • C# 图解教程 第5版 —— 第18章 泛型
  • 保障事务隔离级别的关键措施
  • Docker导入导出镜像、导入导出容器的命令详解以及使用的场景
  • 虚拟化嵌套
  • 【XILINX】记录ISE/Vivado使用过程中遇到的一些warning及解决方案
  • Tableau进阶--Tableau数据故事慧(20)解构Tableau的绘图逻辑
  • 45.0/HTML 简介(详细版)
  • Python 如何进行游戏开发?