到这里,你可以把 Binder 驱动看作一个机器,它连接着所有 Services(假设 app 只调用 Services 提供的接口)。
一个 app(客户端)想要找一个 Service 办点事,它就要去操作这个机器,而这个机器会检测 app 的身份,如果身份合法,则可以继续操作。假设 app 要调用 A 服务的 foo
函数,那么它可以告诉这个机器,这个机器随后就会检查连在它身上的所有 Services,找到 app 需要的那一个,让它执行 foo
函数,并且再将返回值告诉 app,这样 app 就成功隔着进程操作到 A 服务了。
这当然是很抽象的说法,系统在 Framework 层做了很好地封装,让我们可以友好地使用 Binder 驱动来进行 IPC 操作,下面我们就直接看应用层所提供的接口是如何工作的。
探究与 Binder 相关的 Java 类
要说这些类,我们首先要用到它们,最简单的方式就是去创建一个 Service,让它运行在单独的进程中,然后编写 AIDL,实现一个接口,再到 Activity 中去使用这个接口。(注意:本文不介绍 Remote Service 与 AIDL 的相关知识,如果读者对这部分内容还不够了解,请先将它们弄懂再回来看本文)
AIDL 代码生成器为我们创建了一个 java 文件,这里面涉及到几个类:
image
这些就是在应用层使用 Binder 所需的所有类了,看类图有些错综,但实际还是很简单的。
AIDL 生成的就是 IMyService
这个接口,而 Stub
和 Proxy
则是这个接口的两个内部类,分别是 Binder
类和 BinderProxy
类的子类(Proxy
类虽然是用组合方式来持有 BinderProxy
的,但实际就是在直接用这个类,只不过做了一层封装,让其更易使用而已),Stub
和 Proxy
都实现了 IMyService
。
所以 IInterface
到底是什么,它就是一个用于表达 Service
提供的功能的一个契约,也就是说 IInterface
里有的方法,Service
都能提供,调用者你别管用的是 BinderProxy
还是什么,只要拿到 IInterface
,你就可以直接调用里面的方法,它就是一个接口。
同时 Stub
虽然实现了 IMyService
,但是并没有实现厘米的任何方法,它是一个抽象类,开发者需要自己子类化 Stub
去实现具体的功能。
Proxy
实现了 IMyService
,并且实现了里面的方法,这些方法的实现我们下面再说。
为什么 IMyService
要分 Stub
和 Proxy
呢?这是为了要适用于本地调用和远程调用两种情况。如果 Service 运行在同一个进程,那就直接用 Stub
,因为它直接实现了 Service 提供的功能,不需要任何 IPC 过程。如果 Service 运行在其他进程,那客户端使用的就是 Proxy
,这里这个 Proxy
的功能大家应该能想到了吧,就是把参数封装后发送给 Binder 驱动,然后执行一系列 IPC 操作最后再取出结果返回。
好,到这里请求用 Proxy
发出去了,Service 怎么接受请求并作出响应呢,这就需要 Stub
了,还记得我们的 Stub
是继承自 Binder
的吗?Binder
有一个 onTransact
方法,而 Stub
重写了这个函数,这个函数三个重要参数:int code
、Parcel data
、Parcel reply
,分别对应了被调函数编号、参数包、响应包。当 Proxy
发起了一个请求,服务端中相应的响应线程就会通过 JNI 调用到 Stub
类,然后执行里面的 execTransact
方法,进而转到 onTransact
方法。(这一系列调用链大家可以从源码中分析出来,我这里作为抛砖引玉,就不带大家分析 native 层的代码了)
我们来看一眼这个 onTransact
:
@Override
public boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException {
switch (code) {
case INTERFACE_TRANSACTION:
{
reply.writeString(DESCRIPTOR);
return true;
}
case TRANSACTION_increaseCounter:
{
data.enforceInterface(DESCRIPTOR);
int _arg0;
_arg0 = data.readInt();
int _result = this.increaseCounter(_arg0);
reply.writeNoException();
reply.writeInt(_result);
return true;
}
}
return super.onTransact(code, data, reply, flags);
}
是不是非常清晰易懂,就是根据传过来的数据包来做相应的操作,然后把结果写回数据包,Binder 驱动会来帮我们做好这些数据包的分发工作。而这段代码是运行在 Service 本地进程中的,它可以直接调用实现好的 Stub 类中的相关方法(本例子中是 increaseCounter
方法)。
下面我们趁热打铁再来看一眼 Proxy
类中的 increaseCounter
是怎么实现的:
@Override
public int increaseCounter(int increment) throws RemoteException{
Parcel _data = Parcel.obtain();
Parcel _reply = Parcel.obtain();
int _result;
try {
_data.writeInterfaceToken(DESCRIPTOR);
_data.writeInt(increment);
mRemote.transact(Stub.TRANSACTION_increaseCounter, _data, _reply, 0);
_reply.readException();
_result = _reply.readInt();
}
finally {
_reply.recycle();
_data.recycle();
}
return _result;
}
正好与 Stub
中的处理能够对应起来,其实这两段代码就是整个 IPC 的核心了,Binder 驱动和 Binder 类在底层帮我们做好了其他一切事情。
休息一下。
下面我们来思考另一件事情,如何判断 Service 运行在同一进程还是不同进程?
我们知道,Service
有一个 onBind
方法,这里面就返回了我们实现好的 Stub
类,而客户端 bind service 时拿到的又是一个 IBinder
对象,我们每次只需要调用 Stub
的 asInterface
静态方法,把这个 IBinder
对象传进去就能拿到 Stub
类或者 Proxy
类了,看起来十分智能!那么这个 asInterface
又蕴藏什么玄机呢?我们来看一眼实现代码:
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
总结
找工作是个很辛苦的事情,而且一般周期都比较长,有时候既看个人技术,也看运气。第一次找工作,最后的结果虽然不尽如人意,不过收获远比offer大。接下来就是针对自己的不足,好好努力了。
最后为了节约大家的时间,我把我学习所用的资料和面试遇到的问题和答案都整理成了PDF文档
喜欢文章的话请关注、点赞、转发 谢谢!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
是针对自己的不足,好好努力了。
最后为了节约大家的时间,我把我学习所用的资料和面试遇到的问题和答案都整理成了PDF文档
喜欢文章的话请关注、点赞、转发 谢谢!
[外链图片转存中…(img-0LhGjNQF-1712015545713)]