Android中的IPC-(Binder)

最近在重新看<Android开发艺术探索>准备将阅读的记录下来,加深理解

Binder

  1. 从来类的角度来说,Binder就是Android的一个类,它继承了IBinder接口

  2. 从IPC的角度来说,Binder是Android中的一个中的一种跨进程通信方式,Binder还可以理解为一种虚拟的物理设备,它的设备驱动是/dev/binder,该通信方式在Linux中没有(由于耦合性太强,而Linux没有接纳)

  3. 从Android Framework角度来说,Binder是ServiceManager连接各种Manager(ActivityManager、WindowManager等)和相应的ManagerService的桥梁

  4. 从Android应用层的角度来说,Binder是客户端和服务端进行通信的媒介,当你bindService的时候,服务端会返回一个包含了服务端业务调用的Binder对象,通过这个Binder对象,客户端就可以获取服务端提供的服务或者数据,这里的服务包括普通服务和基于AIDL的服务

Android开发中,Binder主要是用于Service中,包括了Aidl和Messenger,Messenger的底层实现其实就是AIDL.

我们新建一个aidl文件,build之后系统会自动根据创建的aidl文件生成一个同名的java文件,他本质是一个继承了android.os.IInterface的接口,所有可以在Binder中传输的接口都需要继承IInterface这个接口.

系统生成的代码中,会有之前在aidl中声明的方法 同时有一个抽象的Stub类,这个Stub其实就是一个Binder类,同时在Stub中还有一个内部Proxy类。

这里我绕了很久才明白为什么在Proxy内的方法是运行在客户端的,onTransact的方法是运行于服务端的,我们这里可以先看一下概念,后续会通过代码进行分析
首先可以查看一下Stub内部的元素:


  • DESCRIPTOR:唯一标识符
  • Stub():构造函数
  • asInterface(android.os.IBinder obj) :用于将远程的Binder对象转换为客户端所需要的AIDL对象.在此函数中,如果服务端和客户端是处于同一进程,返回的就是服务端的Stub对象,如果不是则会返回封装的Stub.Proxy对象
  • asBinder:返回当前的Binder对象
  • onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags):该方法运行与服务端的Binder线程池中,当客户端发起了跨进程的请求时,远程请求会通过系统底层封装后交给该函数进行处理,服务端可以通过code知道远程调用的是哪一个方法,将传入的data中取出方法需要的参数(如果方法需要的话),然后执行目标方法.执行完毕后将返回值放入到reply中,返回false的话客户端的调用会失败
    Proxy类:代理类
    内部有一个android.os.IBinder的内部变量mRemote,代表了远程的Ibinder对象,构造函数为

Proxy(android.os.IBinder remote) {

mRemote = remote;

}

Proxy#声明的方法:

这个方法会运行在客户端,当客户端进行调用时,首先会获取到两个Parcel对象,一个为传入的数据,一个为返回的数据. 通过函数_data.writeInterfaceToken(DESCRIPTOR);将当前远程的唯一描述符写入,接着通过函数transact调用相应的函数方法, 接着函数挂起,等待执行完毕后会读取返回到的数据然后返回.接着我们通过代码来查看一下这个过程.


@Override

public java.util.List<Book> getBookList() throws android.os.RemoteException {

   android.os.Parcel _data = android.os.Parcel.obtain();

    android.os.Parcel _reply = android.os.Parcel.obtain();

    java.util.List<Book> _result;

    try {

       _data.writeInterfaceToken(DESCRIPTOR);

        mRemote.transact(Stub.TRANSACTION_getBookList, _data, _reply, 0);

        _reply.readException();

        _result = _reply.createTypedArrayList(com.hy.bindertest.Book.CREATOR);

    }finally {

        _reply.recycle();

        _data.recycle();

    }

return _result;

}

接着我们进入mRemote.transact()这个函数中.


/**

* Default implementation rewinds the parcels and calls onTransact.  On

* the remote side, transact calls into the binder to do the IPC.

*/

public final boolean transact(int code, @NonNull Parcel data, @Nullable Parcel reply, 
                              int flags)throws RemoteException {

    if (false) Log.v("Binder", "Transact: " + code +" to " +this);

    if (data !=null) {

        data.setDataPosition(0);

    }

    boolean r = onTransact(code, data, reply, flags);//这里代用了onTransact方法

    if (reply !=null) {

        reply.setDataPosition(0);

    }
 
    return r;

}

会发现他最后调用的就是mRemote中的onTransact函数,而mRemote这个远程Binder对象,是运行于服务端的BInder线程中,所以这里也就是所说的onTransact方法运行于服务端的Binder线程池中的原因,这时候当前线程会被挂起,等待RPC过程结束后,当前线程继续进行,然后从_replay中获取到RPC过程返回的结果,最后返回_replay中的数据.

所以,我们就知道,当客户端发起一次远程的调用时,当前的线程会被挂起知道服务端返回数据,所以一个远程函数是很耗时的,我们不能在UI线程中进行调用,而服务器端的Binder方法是运行于Binder的线程池中的,所以Binder中的方法我们不需要再开线程进行执行.

所以整个过程就比较清晰了:

首先客户端发起一个远程的请求->Binder收到之后,将参数写入到data中,调用transact方法执行,远程的Service中的OnTransact函数指定函数被调用,将结果写入到reply返回到Binder中,Binder获取到结果,返回给Client;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容