Android Framework实战开发-Binder专题讲解之aidl文件的详细分析

107 篇文章 22 订阅
23 篇文章 0 订阅

csdn在线学习课程,课程咨询答疑和新课信息:QQ交流群:422901085进行课程讨论

android跨进程通信实战视频课程(加群获取优惠)

大家平时做应用开发时候也经常会遇到有跨进程通信的需求,这里大部分通信可能就会回答,就是用aidl,service。确实这个aidl和service的方式是应用开发中对binder接触层面应该属于最为接近的一层。因为其他的接口方式跨进程通信,比如startActivtiy,ContentProvider,broadcast,这些都是系统组件接口直接帮我们做好的,连binder的影子都看不见,自然也就对binder相对比较陌生。
本节就来重点介绍一下深入讲解分析应用层面binder通信的方式及原理,这一部分属于binder讲解的中级部分,这节课来讲解一下binder的aidl生成java文件的源码分析:
为什么要AIDL,它存在意义是?
在使用binder service进行跨进程通信经常我们会写aidl,那首先大家肯定会有疑问,那aidl到底是什么?AIDL是一个缩写,全称是Android Interface Definition Language,也就是Android接口定义语言。所以说aidl他只是一个语言,主要目的就是来帮助binder跨进程通信的接口定义而设计的一个语言,后缀是 .aidl结尾,他有自己的语法,不过基本和java比较类似,最后会通过一个aidl.exe把整个aidl文件都转换成对应的java文件,因为编译接口啥事我们java程序在使用当然要变成java 修改类文件,那这里同学肯定会说为什么不直接用java,还要搞个新语言,其实这里就问到了最关键的部分。
这里有以下原因:
1、如果要java语言实现那会导致要写很多重复的固定代码,写的代码量比较大,稍微不小心容易出错
2、这一部分完全可以让脚本根据某些配置文件来生成对应的java代码,而这个配置文件的正类似我们今天看到的aidl
本质如下图所示:
在这里插入图片描述
这里我们来按照上图来一步步实现:
1、明确哪个是客户端,哪个是服务
首先应用层跨进程通信时候,我们都一般会定义一个Service,因为Android中Service组件是可以运行于独立的进程即Service代码可能是在某一个apk下面,但是它运行的进程却不是这个apk的包名对应的进程,所以一般演示跨进程通信完全可以一个apk实现,这个Service就是我们真正的服务端,也是真正的业务实现端,客户端就是调用这个Service发起端,这个具体是在Activity中实现还是Service一般都是可以,一般我们为了方便就在Activity中进行跨进程通信的调用,即Activity作为客户端。
2、确定通信过程中的接口(即通信要获取的信息)
这个就是所说的aidl文件,它来定义好对应的通信接口接口,比如这里我们定义一个学生信息获取接口:
IStudentInterface .aidl

interface IStudentInterface {
    int getStudentId(String name);//定义一个根据学生名字查询学号的接口
}

aidl文件的转换
这个aidl文件写好后,点击make编译一下工程,在编译过程就会被aidl.exe转成对应的java文件,具体查看这个java文件,可以在android studio点击如下路径:
在这里插入图片描述
如果想要直接使用aidl.exe来命令生成也是完全可以
先到安卓sdk\build-tools\29.0.2下有aidl.exe

aidl.exe -IF:\binder_drivers_code\ServiceDemo\app\src\main\aidl\com\example\servicedemo  F:\binder_drivers_code\ServiceDemo\app\src\main\aidl\com\example\servicedemo\IStudentInterface.aidl

"-I"与"F:\binder_drivers_code***"之间是没有空格的,最后就会在F:\binder_drivers_code\ServiceDemo\app\src\main\aidl\com\example\servicedemo生成一个IStudentInterface.java
在这里插入图片描述
这里相信大家已经明白了aidl是怎么一回事,它的本质目的就是为了帮助我们减少编写那些重复不变化的通信协议代码,这个完全可以让机器根据我们的描述配置文件来生成,这个描述配置文件其实就是我们的aidl文件

3、那我们已经生成了这个java文件,那我这个java文件到底内容是什么呢?
其实上面的图也提前展示了,它主要分别一部分是Stub 类和Stub .Proxy类

  public static abstract class Stub extends android.os.Binder implements com.example.servicedemo.IStudentInterface1
  {
       。。省略
    public static com.example.servicedemo.IStudentInterface1 asInterface(android.os.IBinder obj)
    {
         。。省略
      return new com.example.servicedemo.IStudentInterface1.Stub.Proxy(obj);
    }
    @Override public android.os.IBinder asBinder()
    {
      return this;
    }
    //服务端,这个时候服务端驱动获取数据后,一系列调用会回调到onTransact
    @Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException
    {
      java.lang.String descriptor = DESCRIPTOR;
      switch (code)
      {
           。。省略
        case TRANSACTION_getStudentId:
        {
          data.enforceInterface(descriptor);
          java.lang.String _arg0;
          _arg0 = data.readString();
          int _result = this.getStudentId(_arg0);//这个地方就会真正调用Service中实现的那个getStudentId方法
          reply.writeNoException();
          reply.writeInt(_result);
          return true;
        }
            。。省略
      }
    }
    private static class Proxy implements com.example.servicedemo.IStudentInterface1
    {
     。。省略
      @Override public int getStudentId(java.lang.String name) throws android.os.RemoteException
      {
        android.os.Parcel _data = android.os.Parcel.obtain();
        android.os.Parcel _reply = android.os.Parcel.obtain();
        int _result;
        try {
          _data.writeInterfaceToken(DESCRIPTOR);
          _data.writeString(name);
          boolean _status = mRemote.transact(Stub.TRANSACTION_getStudentId, _data, _reply, 0);//客户端调用getStudent最后是通过mRemote调用到远程,并等待获取结果
          if (!_status && getDefaultImpl() != null) {
            return getDefaultImpl().getStudentId(name);
          }
          _reply.readException();
          _result = _reply.readInt();
        }
        finally {
          _reply.recycle();
          _data.recycle();
        }
        return _result;
      }
      public static com.example.servicedemo.IStudentInterface1 sDefaultImpl;
    }
       。。省略
  }

这里Stub就是远程端的具体真实实现,一般在服务端Serivce中实现,而Stub.Proxy类则是主要给客户端提供调用远程端的接口。对应的实际使用Stub类和Stub.Proxy类如下截图:
Stub的服务端实现截图:
在这里插入图片描述
Stub.Proxy的具体使用调用过程:
在这里插入图片描述
这里的asInterface上面代码也展示,其实就是
new com.example.servicedemo.IStudentInterface1.Stub.Proxy(obj);构造了一个Stub.Proxy本地对象。
最后总结一下aidl转成java文件后看到的一个跨进程调用的图:
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
千里马8年Android系统及应用开发经验,曾担任过美国unokiwi公司移动端技术总监兼架构师,对系统开发,性能优化,应用高级开发有深入的研究,Android开源定制ROM Lineage的贡献者之一,国内首家线下开辟培训Android Framework课程,拥有2年的Android系统培训经验。成为腾讯课堂专业负责android framework课程分享第一人,致力于提高国内android Framework水平Android Framework领域内是国内各大手机终端科技公司需要的人才,应用开发者都对Android系统充满着好奇,其中的binder是重中之重,都说无binder无Android,binde是Android系统的任督二脉。课程水平循序渐进,由中级再到高级,满足各个层次水平的android开发者。1、灵活使用binder跨进程通信,在app端对它的任何api方法等使用自如2、可以单独分析android系统源码中任何binder部分,分析再也没有难度3、掌握binder驱动本质原理,及对应binder驱动怎么进行跨进程通信,及内存等拷贝方式数据等4、对binder从上层的java app端一直到最底层的内核binder驱动,都可以顺利理通5、针对系统开发过程中遇到的binder报错等分析方法,及binder bug案例学习6、针对面试官任何的binder问题都可以对答自如7、socket这种跨进程通信实战使用8、针对android源码中使用的socket源码轻松掌握9、android系统源码中最常见的socketpair中双向跨进程通信10、使用socket实现一个可以让app执行shell命令的程序

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值