[进阶] Binder理解和源码解析

基础知识

Android进程间的通讯没有沿用Linux的原有的通讯模式,而是采用新的通讯模式Binder.

先了解下Linux原有的通讯方式:Linux进程间的通信方式和原理

进程通信的应用场景一般包含数据/资源共享,数据传输,通知事件,进程控制。主动被动都有了。

Linux原有的几种通信方式

  • pipe管道
  • 消息队列(Message)
  • 信号量(Semaphore)
  • 共享内存(Share Memory)
  • 信号(Signal)
  • 跟踪(Trace)
  • 还有支持支持C/S的Socket

Android为什么采用Binder?

Android没有使用Linux原生的通信机制,而是使用Binder机制。
Binder机制是采用OpenBinder演化而来,采用C/S的通信模式。
在Android中使用它的原因如下

  • socket效率低,开销大。linux目前只有socket支持C/S
  • 管道和消息队列效率低。管道和消息采用存储转发方式,所以至少需要拷贝2次数据,效率低
  • 共享内存控制机制复杂:共享内存虽然在传输时没有拷贝数据,但其控制机制复杂(比如跨进程通信时,需获取对方进程的pid,得多种机制协同操作)。
  • Binder安全性更高
    • Linux的IPC机制在本身的实现中,并没有安全措施,得依赖上层协议来进行安全控制。
    • Binder机制的UID/PID是由Binder机制本身在内核空间添加身份标识,安全性高;并且Binder可以建立私有通道,这是linux的通信机制所无法实现的(Linux访问的接入点是开放的)

各种IPC方式数据拷贝次数对比

IPC方式 数据拷贝次数
共享内存 0
Binder 1
Socket/管道/消息队列 2

总之,基于以上原因,Linux上原生机制无法满足手机的要求,Android需要建立一套新的IPC机制来满足系统对通信方式的传输性能和安全性要求,这就出现了Binder

Binder是什么

Binder是Android系统中进程间通讯(IPC)的一种方式,像Intent, Messenger, ContentProvider底层都是Binder实现

  • Binder指的是一种通信机制。AIDL使用Binder进行通信,指的就是Binder这种IPC机制。
  • 对于Server进程来说,Binder指的是Binder本地对象。

    • Server端一般会实例一个Binder实现具体的接口,即Server能够提供的能力(注意因为Stub是abstract不能直接实例化,实例化的父类Binder对象)
    • 在onBind时将实例传递给Client

      ```
      public class BookManagerService extends Service {
          private static final String TAG = BookManagerService.class.getSimpleName();
      
          // book list
          private CopyOnWriteArrayList<Book> mBookList = new CopyOnWriteArrayList<Book>();
          // new book listener RemoteCallbackList used for multi-process listener
          private RemoteCallbackList<INewBookListener> mListenerList = new RemoteCallbackList<INewBookListener>();
          private Binder mBinder = new IBookManager.Stub() {
      
              @Override
              public List<Book> getBookList() throws RemoteException {
                  SystemClock.sleep(2000);
                  return mBookList;
              }
      
              @Override
              public void addBook(Book book) throws RemoteException {
                  mBookList.add(book);
                  onNewBookArrived(book);
              }
          ......
          }
      
          @Override
          public IBinder onBind(Intent intent) {
              if (!verifyPermission()) {
                  return null;
              }
              return mBinder;
          }
      }
      
      ```
      
  • 对于Client来说,Binder指的是Binder代理对象,只是Binder本地对象的一个远程代理

    • Client通过asInterface方法获取Proxy代理或通过asBinder方法获取Binder实例
    • 对这个Binder代理对象的操作,会通过驱动最终转发到Binder本地对象上去完成,即调用Service的Binder实例的方法。
    • 对于一个拥有Binder对象的使用者而言,它无须关心这是一个Binder代理对象还是Binder本地对象。对于代理对象的操作和对本地对象的操作对它来说没有区别。因为实现了同一个接口
        public class BookManagerActivity extends AppCompatActivity {
            private static final String TAG = BookManagerActivity.class.getSimpleName();
    
            private IBookManager mRemoteBookManager;
    
            @Override
            protected void onCreate(Bundle savedInstanceState) {
                super.onCreate(savedInstanceState);
                setContentView(R.layout.activity_book_manager);
                initView();
    
                // bind service : after 5.0 must follow
                // https://developer.android.com/google/play/billing/billing_integrate.html#billing-requests
                Intent intent = new Intent(this, BookManagerService.class);
                intent.setPackage(getPackageName());
                bindService(intent, mServiceConn, Context.BIND_AUTO_CREATE);
            }
    
            private ServiceConnection mServiceConn = new ServiceConnection() {
                public void onServiceConnected(ComponentName className, IBinder service) {
                    mRemoteBookManager = IBookManager.Stub.asInterface(service);
                    if (mRemoteBookManager != null)
                        try {
                            mRemoteBookManager.asBinder().linkToDeath(mDeathRecipient, 0);
                            mRemoteBookManager.registerListener(mNewBookListener);
                        } catch (RemoteException e) {
                            e.printStackTrace();
                        }
                    Log.d(TAG, "bind service");
                }
    
                public void onServiceDisconnected(ComponentName className) {
                    mRemoteBookManager = null;
                    Log.d(TAG, "onServiceDisconnected, tid = " + Thread.currentThread().getName());
                }
            };
    
            public void addBook(View v) {
                if (mRemoteBookManager == null) {
                    Log.w(TAG, "mRemoteBookManager is null");
                    return;
                }
    
                bookThreadPool.execute(new Runnable() {
                    @Override
                    public void run() {
                        try {
                            int bookId = mRemoteBookManager.getBookSize() + 1;
                            Book newBook = new Book(bookId, "new-" + bookId);
                            mRemoteBookManager.addBook(newBook);
                            Log.d(TAG, "add book:" + newBook);
                        } catch (RemoteException e) {
                            e.printStackTrace();
                        }
                    }
                });
            }
        }
    
  • 对于传输过程而言,Binder是可以进行跨进程传递的对象;Binder驱动会对具有跨进程传递能力的对象做特殊处理:自动完成代理对象和本地对象的转换

代码片一

interface IBookManager {
    // get all book list
    List<Book> getBookList();
    // add one book
    void addBook(in Book book);
    // get book list size
    int getBookSize();
    // register and unregister new book listener
    void registerListener(INewBookListener listener);
    void unregisterListener(INewBookListener listener);
}

编译器生成的IBookManager

这里写图片描述

全部Code在这里Github IPC

Binder架构

Binder架构图(来自网上)

架构图

详细架构图(来自网上)

这里写图片描述

  • Binder 通信采用 C/S 架构:包含 Client、 Server、 ServiceManager 以及 Binder 驱动。
  • ServiceManager 用于管理系统中的各种服务。无论是注册服务和获取服务的过程都需要ServiceManager(Native C++层的ServiceManager不是framework java层的ServiceManager
  • Binder 在 framework 层进行了封装,通过 JNI 技术调用 Native(C/C++)层的 Binder 架构。
  • Binder 在 Native 层以 ioctl 的方式与 Binder 驱动通讯

Binder机制

下图是Binder机制是如何跨进程。(来自网络)

每个Android的进程,只能运行在自己进程所拥有的虚拟地址空间。对应一个4GB的虚拟地址空间,其中3GB是用户空间,1GB是内核空间(大小可以通过参数配置调整)。对于用户空间,不同进程之间彼此是不能共享的,而内核空间却是可共享的。Client进程向Server进程通信,恰恰是利用进程间可共享的内核内存空间来完成底层通信工作。Client端与Server端进程往往采用ioctl等方法跟内核空间的驱动进行交互。

ServiceManager是整个Binder通信机制的大管家,是Android进程间通信机制Binder的守护进程。当Service Manager启动之后,Client端和Server端通信时都需要先获取Service Manager接口,才能开始通信服务。ServiceManager是由init进程通过解析init.rc文件而创建的。

  • 注册服务(addService):Server进程要先注册Service到ServiceManager。该过程:Server是客户端,ServiceManager是服务端。
  • 获取服务(getService):Client进程使用某个Service前,须先向ServiceManager中获取相应的Service。该过程:Client是客户端,ServiceManager是服务端。
  • 使用服务:Client根据得到的Service信息建立与Service所在的Server进程通信的通路,然后就可以直接与Service交互。该过程:client是客户端,server是服务端。

这里写图片描述

Binder通信的基本流程

一次完整Binder通信图(来自网上)

这里写图片描述

深入理解Java层的Binder

  • IBinder是一个接口,它代表了一种跨进程传输的能力
    • 只要实现了这个接口,就能将这个对象进行跨进程传递。这是驱动底层支持的
    • 在跨进程数据流经驱动的时候,驱动会识别IBinder类型的数据,从而自动完成不同进程Binder本地对象以及Binder代理对象的转换
  • IBinder负责数据传输,IInterface代表的就是远程server对象具有什么能力。就是aidl里面的接口
  • Java层的Binder类,代表的其实就是Binder本地对象。
    • BinderProxy类是Binder类的一个内部类,它代表远程进程的Binder对象的本地代理;这两个类都继承自IBinder, 因而都具有跨进程传输的能力;
    • 实际上,在跨越进程的时候,Binder驱动会自动完成这两个对象的转换。
  • 在使用AIDL的时候,编译工具会给我们生成一个Stub的静态内部类;
    • 继承了Binder, 说明它是一个Binder本地对象
    • 它实现了IInterface接口,表明它具有远程Server承诺给Client的能力;
    • Stub是一个抽象类,具体的IInterface的相关实现需要我们手动完成,这里使用了策略模式。

这篇的深入理解Java层的Binder

Binder源码解析

framework层的源码分析可以看这里

Reference

Android为什么选择binder

一篇文章了解相见恨晚的 Android Binder 进程间通讯机制

Binder学习指南

老罗的 Android进程间通信(IPC)机制Binder简要介绍和学习计划 系列

Innost的 深入理解Binder 系列

Gityuan的 Binder系列 (基于 Android 6.0) Binder系列—开篇

Android Binder机制原理

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章