Android源码代理模式---Binder

Binder是Android的进程间通信核心,如果看过Android源码,你会发现源码中Android的各种核心服务都是通过Binder机制进行相互通信的。在Binder的client部分就是通过代理模式来访问Server端的。这里想通过代理模式来详细介绍Java层Binder。文中会简单介绍代理模式,详细介绍Binder机制。(源码基于6.0.1)

代理模式

意图

对其他对象提供一种代理以控制对这个对象的访问。

UML图

代码示例

    abstract class Subject{
        public abstract void operate();
    }
    class RealSubject{
        public void operate(){
            System.out.println("real operate");
        }
    }
    class Proxy{
        private Subject realSubject;
        Proxy(Subject subject){
            realSubject = subject;
        }
        public void operate(){
            realSubject.operate();
        }
    }

    public static void main(String[] args){
        RealSubject real = new RealSubject();
        Proxy proxy = new Proxy(real);
        proxy.operate(); //实际上交给了realsubject去操作了。
    }

关于代理模式

上面的代码跟UML图只是一个例子,并不能完全描述代理模式的。代理模式强调的是一种代理,上面的例子中其实直接用RealSubject就可以了,完全没必要用Proxy。另外代理与被代理对象并不要求有共同的父类/接口。代理模式有几种常见的情况:

- 远程代理(Remote Proxy): 如果某个对象无法实例化,不在同一个地址空间,需要通过编码来进行通信。比如需要访问网络服务器上面的一个对象操作。比如我们接下来需要介绍的Binder。

- 虚代理(Virtual Proxy): 在需要的时候创建大对象,比如超大图片,我们可以使用一个虚代理代理图像,在真正需要的时候再去将图像完全加载出来,在这之前只需要在代理里面保存图像的大小,让它有个占位就好了。

- 保护代理(Protect Proxy): 需要对对象的某些操作进行隐藏,那么就可以使用代理对它的接口进行隐藏。

- 智能指引(Smart Reference): 当需要对对象的引用进行计数的时候,可以使用智能指引的代理模式。

Binder

Binder是一个接口形式的IPC。在这里以我们在应用程序开发过程中使用的.aidl文件声明生成的接口为例。使用aidl需要说明的是Android官方文档上面强调过:

使用aidl的必要条件是你的Service想要被其他客户端从不同的应用程序为了IPC而调用,并且想要在你的Service中处理多线程。如果你不需要并行来自不同的应用程序的IPC,那么你只需要使用实现Binder就好了,如果你不需要处理多线程的并行,那么你使用Messenger就好了。

这里只是为了说明Binder,名称为ITestService.aidl:

package com.houzhi.testproject;

interface ITestService{
    String getContent();
}

在编译运行前,Android ide(eclipse/android studio) 会自动生成相关的类,ITestService.Stub, ITestService.Proxy, 而我们通过调用bindService就可以得到Binder的代理。

Intent service; //只是为了表明类型
ServiceConnection connection = new ServiceConnection(){
    public void onServiceConnected(ComponetName name, IBinder service){
        ITestService testService = ITestService.stub.asInterface(service);
        //testService就是我们得到的代理
    }
    public void onServiceDisConnected(ComponentName name){

    }
}
context.bindService(service,connection);

整个自动生成的类以及Binder, IInterface的UML图如下:

ITestService.stub.asInterface(service)的内部就是创建了一个Proxy对象,其实现代码如下:

    public static com.houzhi.testproject.IMyAidlInterface asInterface(android.os.IBinder obj)     {
        if ((obj == null)) {
            return null;
        }
        android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
        if (((iin != null) && (iin instanceof com.houzhi.testproject.IMyAidlInterface))) {
            return ((com.houzhi.testproject.IMyAidlInterface) iin);
        }
        return new com.houzhi.testproject.IMyAidlInterface.Stub.Proxy(obj);
    }

上面的代码需要说明的是obj的queryLocalInterface会返回null, 因为obj其实是一个BinderProxy类型,而BinderProxy类中,并没有给DESCRIPTOR添加对应的IInterface。在源码中具体的实现在native层,native通过jni的方式利用mObject(相当于对应的对象id,BinderProxy的成员变量)创建了一个BinderProxy。将这个BinderProxy返回给客户端。

从UML图就已经看出Binder是一个非常典型的代理模式,是一种远程代理,实际上Proxy代理的是另外一个进程中的Stub对象。内部是将接口函数标记为对应的ID,然后根据这个ID来标识目前调用的是哪一个函数。由DESCRIPTOR来作为Token分隔不同的接口调用,另外通过Parcel来写入函数参数和接受函数返回值(Stub端对应接受参数和写入结果)。aidl可接受的类型也是普通类型(int,double…),以及Map,List,String,

CharSequence,Parcel,和Binder类型。对应的处理也是不一样的,不过终归可以使用基本的方式搞定。

下面是Stub与Proxy关于接口函数实现的具体代码:

public static abstract class Stub extends android.os.Binder implements com.houzhi.testproject.ITestService {
        private static final java.lang.String DESCRIPTOR = "com.houzhi.testproject.ITestService";
        //省略了部分代码
        @Override
        public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException {
            switch (code) {
                case INTERFACE_TRANSACTION: {
                    reply.writeString(DESCRIPTOR);
                    return true;
                }
                case TRANSACTION_getContent: {
                    data.enforceInterface(DESCRIPTOR);
                    java.lang.String _result = this.getContent();
                    reply.writeNoException();
                    reply.writeString(_result);
                    return true;
                }
            }
            return super.onTransact(code, data, reply, flags);
        }

        private static class Proxy implements com.houzhi.testproject.ITestService {
            private android.os.IBinder mRemote;

            Proxy(android.os.IBinder remote) {
                mRemote = remote;
            }
            //省略了部分代码
            @Override
            public java.lang.String getContent() throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                java.lang.String _result;
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    mRemote.transact(Stub.TRANSACTION_getContent, _data, _reply, 0);
                    _reply.readException();
                    _result = _reply.readString();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
                return _result;
            }
        }

        static final int TRANSACTION_getContent = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
    }

Binder底层实现简述

看完上一节其实就能想到怎么去实现Binder了,将接口调用请求转换成数据流的形式,通过int类型的id标识具体是哪一个函数,用token来划分每一次函数调用请求,按照函数参数顺序和类型一个一个地写入和读取参数。而在底层通过一个进程通信(Pipe,共享内存)把这些消息发到服务端/客户端,这样就可以大致实现了。实际上最底层使用的是Binder驱动,服务端通过IPCThreadState开启一个线程(IPCThreadState::self()->startThreadPool()),并且将线程放入到线程池中(IPCThreadState::self()->joinThreadPool()),等待接收来自客户端的请求(talkWithDriver())。而客户端通过BinderProxy(内部保存了本地BpBinder的指针)的transact,最后访问Binder(talkWithDriver())。内部使用访问驱动函数都是ioctl。

ServiceManager

上面大致介绍了服务端,客户端与Binder驱动通信流程。但是大家有没有想过,在客户端需要请求服务端的时候,我怎么知道我究竟想要请求哪一个服务端呢?ServiceManager。ServiceManager是保存了所有的Service的fd(驱动文件号)。通过请求ServiceManager的getService(String name)就可以获取对应的服务。而ServiceManager也是一个服务,它的fd是0,是服务中心管理器。客户端指定name,通过Binder请求ServiceManager,然后得到客户端想要的服务BinderProxy,然后就可以请求服务端了。当然创建服务的时候,首先需要通过ServiceManager.addService将服务添加到ServiceManager中。

Binder连接的几大模块

Android中应用层常用的几个重要的模块都是由Binder联系起来的,ActivityManagerService,WindowManagerService,InputManagerService。下面是他们之间的关系图:

另外还有很多服务,比如说Wifi,NetworkPolicy,Power等等。他们都是在SystemServiceRegistry中注册的,这个注册本来在ContextImpl中,现在移到了一个单独的类SystemServiceRegistry。

代理模式在Binder中的应用分析

代理模式在这里使用的基本上是天衣无缝了,我觉得这是代理模式的经典使用。我们在客户端无法直接访问服务端(因为跨进程,地址空间都不一致),通过代理模式,能够让我们在客户端感觉像是直接访问服务端一样。


如果错过了太阳时你流了眼泪,那你也要错过群星了

时间: 2024-10-07 17:24:33

Android源码代理模式---Binder的相关文章

访何红辉:谈谈Android源码中的设计模式

最近Android 6.0版本的源代码开放下载,刚好分析Android源码的技术书籍<Android源码设计模式解析与实战>上市,我们邀请到它的作者何红辉,来谈谈Android源码中的设计模式,以及近期Android开发中的一些热点话题. 受访嘉宾介绍: 何红辉(@MrSimp1e),前友盟Android工程师,活跃于国内各大技术社区,热爱开源,热爱技术,热爱分享.Android开源库 AndroidEventBus . Colorful 作者, 开发技术前线 站长,<Android源码

Android源码分析之Builder模式

http://www.w3c.com.cn/android%E6%BA%90%E7%A0%81%E5%88%86%E6%9E%90%E4%B9%8Bbuilder%E6%A8%A1%E5%BC%8F Android源码分析之Builder模式

Android源码分析之模板方法模式

模式的定义 定义一个操作中的算法的框架,而将一些步骤延迟到子类中.使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤. 使用场景 1.多个子类有公有的方法,并且逻辑基本相同时. 2.重要.复杂的算法,可以把核心算法设计为模板方法,周边的相关细节功能则由各个子类实现. 3.重构时,模板方法模式是一个经常使用的模式,把相同的代码抽取到父类中,然后通过钩子函数约束其行为. UML类图 角色介绍 AbstractClass : 抽象类,定义了一套算法框架. ConcreteClass1 :

Android源码分析之原型模式

模式的定义 用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象. 使用场景 1.类初始化需要消化非常多的资源,这个资源包括数据.硬件资源等,通过原型拷贝避免这些消耗: 2.通过 new 产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式: 3.一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用,即保护性拷贝. UML类图 角色介绍 Client  :  客户端用户. Prototype : 抽象类或者接口,声明

谈谈23种设计模式在Android源码及项目中的应用

本文首发于个人博客:Lam's Blog - 谈谈23种设计模式在Android源码及项目中的应用,文章由MarkDown语法编写,可能不同平台渲染效果不一,如果有存在排版错误图片无法显示等问题,烦请移至个人博客,如果个人博客无法访问可以留言告诉我,转载请声明个人博客出处,谢谢. 前言 本文将结合实际谈谈23种设计模式,每种设计模式涉及 * 定义:抽象化的定义与通俗的描述,尽量说明清楚其含义与应用场景 * 示例:如果项目中有使用过该模式,则会给出项目中的代码,否则会给出尽可能简单好理解的java

Android源码中的抽象工厂---IPolicy

抽象工厂应用是很广的,在Android源码中,这个IPolicy就是一个简单的抽象工厂模式.下面分析一下IPolicy及其实现,以及创建的相关对象(源码基于5.0.0). 抽象工厂 意图 提供一个创建一系列相关或相互依赖对象的接口,而无需指定他们具体的类. UML类图 通过继承抽象工厂,可以产生不同的产品系列 代码示例 abstract class AbsFactory{ public abstract AbsProductA createProduct1(); public abstract

Activity生命周期的回调,你应该知道得更多!--Android源码剖析(上)

转载请标明原文地址:http://blog.csdn.net/yalinfendou/article/details/46909173[yalinfendou的博客] 学习Android近一年,最近几天总算把Activity启动的生命周期回调流程走通了,因为所涉及的知识点太多,赶快做了笔记,不然过几天就忘了.强烈推荐<Android内核剖析>这本书,虽然随着Android版本的不断迭代更新,代码变化了不少,仍具有很强的参考价值. 本篇所涉及知识点: Android 从开机到第一个Activit

Android 源码系列之&lt;十一&gt;从源码的角度深入理解AccessibilityService,打造自己的APP小外挂(下)

转载请注明出处:http://blog.csdn.net/llew2011/article/details/52843637 在上篇文章Android 源码系列之<十>从源码的角度深入理解AccessibilityService,打造自己的APP小外挂(上)中我们讲解了通过AccessibilityService实现自动安装APK小外挂的操作流程,如果你还没有看过上篇文章请点击这里.在这篇文章中我将带领小伙伴从源码的角度来深入学习一下AccessibilityServie的技术实现原理,希望这

《Android源码设计模式解析与实战》读书笔记(十八)

第十八章.代理模式 代理模式也称委托模式,是结构型设计模式之一.是应用广泛的模式之一. 1.定义 为其他对象提供一种代理以控制对这个对象的访问. 2.使用场景 当无法或不想直接访问某个对象或访问某个对象存在困难时可以通过一个代理对象来间接访问,为了保证客户端使用的透明性,委托对象与代理对象需要实现相同的接口. 3.UML类图 (1)Subject:抽象主题类,声明真实主题与共同接口方法,该类可以是抽象类或接口. (2)RealSubject:真实主题类(被委托类),尤其执行具体的业务逻辑方法.