安卓使用 AIDL 实现进程间通信 IPC

AIDL (Android Interface Definition Language) 是一种IDL 语言,用于生成可以在Android设备上两个进程之间进行进程间通信(inter process communication, IPC)的代码。如果在一个进程中(例如Activity)要调用另一个进程中(例如Service)对象的操作,就可以使用 AIDL 生成可序列化的参数。

IPC 是进程间通信的意思,其主要方式有以下几种:

  • Bundle
  • 文件共享
  • AIDL
  • Messenger
  • ContentProvider
  • Socket

一 、IPC 的基础和概念

1.多进程模式

a. 进程与线程

  • 进程:一般指一个执行单元,在 PC 和移动设备上指一个程序应用
  • 线程:CPU 调度的最小单元,线程是一种有限的系统资源。

两者关系:一个进程可包含多个线程,即一个应用程序上可以执行多个任务。

  • 主线程(UI 线程):UI 操作
  • 有限个子线程:耗时操作

b. 开启多进程的方式:

  • (不常用)通过 JNI 在 native 层 fork 一个新的进程
  • (常用) 对安卓四大组件的配置清单页面设置 android:precess,进程名的命名规则:
  1. 默认进程,即省略不写,默认是在当前进程 android:process = "com.test.applicatio”
  2. 以: 开头的进程,通过使用 android:process = " :remote" 方式,全称是 android:process = "com.test.application:remote",表示是当前应用的私有进程,其他进程的组件不能和它跑在同一进程中
  3. 以 . 开头的全局进程,android:process = " .remote" ,全称是 android:process = "com.test.application.remote",其他应用可以通过 ShareUID 的方式和它跑在同一进程中。

UID&ShareUID:
Android系统为每个应用分配一个唯一的UID,具有相同UID的应用才能共享数据。
两个应用通过ShareUID跑在同一进程的条件:ShareUID相同且签名也相同。
满足上述条件的两个应用,无论是否跑在同一进程,它们可共享data目录,组件信息。
若跑在同一进程,它们除了可共享data目录、组件信息,还可共享内存数据。它们就像是一个应用的两个部分。

如何设置 ShareUid 相同呢,很简单:

**//第一个应用程序为的menifest文件代码如下:**
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
package="com.mythou.serviceID"
android:sharedUserId="com.mythou.share">
//.......
**//第二个应用程序的menifest文件代码如下:**
 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
 package="com.mythou.clientID" 
 android:sharedUserId="com.mythou.share">
  1. IPC 简介

a. IPC(Inter-Process Communication,跨进程通信):指两个进程之间进行数据交换的过程。

b. 任何一个操作系统都有对应的IPC机制。
Windows:通过剪切板、管道、油槽等进行进程间通讯。
Linux:通过命名空间、共享内容、信号量等进行进程间通讯。
Android:没有完全继承Linux,比如,其独具特色的通讯方式有Binder、Socket等等。

c. IPC的使用场景:
由于某些原因,应用自身需要采用多进程模式来实现。可能原因有:
某些模块因特殊原因要运行在单独进程中;
为加大一个应用可使用的内存,需通过多进程来获取多份内存空间。
当前应用需要向其它应用获取数据。

d. Android的进程架构:每一个Android进程都是独立的,且都由两部分组成,一部分是用户空间,另一部分是内核空间。

e. Binder 工作原理:
服务器端:在服务端创建好了一个Binder对象后,内部就会开启一个线程用于接收Binder驱动发送的消息,收到消息后会执行onTranscat(),并按照参数执行不同的服务端代码。

Binder驱动:在服务端成功创建 Binder 对象后,Binder驱动也会创建一个mRemote对象(也是Binder类),客户端可借助它调用transcat()即可向服务端发送消息。

客户端:客户端要想访问Binder的远程服务,就必须获取远程服务的Binder对象在Binder驱动层对应的mRemote引用。当获取到mRemote对象的引用后,就可以调用相应Binder对象的暴露给客户端的方法。

如何使用 AIDL 进行 IPC(进程间通信)

  1. 先建立一个安卓工程作为跨进程通信的服务端。

新建一个包名,用来存放 aidl 文件,例如 com.example.aidl ,在里边新建一个 IMyService.aidl 接口文件,文件里定义两个方法用来返回数据和添加数据,这里的数据使用了自定义对象,所以还要建立对象的 aidl 文件和对应的 java 文件,并且要和 IMyAidlInterface.aidl 放在同一个包名下。

IMyAidlInterface.aidl 代码如下:

//IMyAidlInterface.aidl
package com.example.myapplication;
import com.example.myapplication.PersonBean;

// Declare any non-default types here with import statements​interface 
IMyAidlInterface {
    void addPerson(in PersonBean mBean);
    List<PersonBean> getPerson();
}

aidl 中支持的参数类型有:基本类型(int,long,char,boolean等),String,CharSequence,List,Map,其他类型必须使用 import 导入即使它们可能在同一个包里,比如上面的 PersonBean,尽管它和IMyService在同一个包中,但是还是需要显示的 import 进来。另外,接口中的参数除了 aidl 支持的类型,其他类型必须标识其方向:到底是输入还是输出抑或两者兼之,用 in,out 或者 inout 来表示,上面的代码我们用in标记,因为它是输入型参数。

PersonBean.aidl 文件代码如下,比较简单就是一个当前包的路径和一个定义

package com.example.myapplication;parcelable PersonBean;

这里的 PersonBean 必须用 parcelable 定义,这里的 parcelable 和序列化的 Parcelable 不是同一个东西,这里只是标记一个类型。

  1. 然后是 PersonBean.java 类:
package com.example.myapplication;

import android.os.Parcel;
import android.os.Parcelable;

public class PersonBean implements Parcelable {
    private int age;
    private String name;

    public PersonBean(int age, String name) {
        this.age = age;
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }


    protected PersonBean(Parcel in) {
        age = in.readInt();
        name = in.readString();
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(age);
        dest.writeString(name);
    }

    @Override
    public int describeContents() {
        return 0;
    }

    public static final Creator<PersonBean> CREATOR = new Creator<PersonBean>() {
        @Override
        public PersonBean createFromParcel(Parcel in) {
            return new PersonBean(in);
        }

        @Override
        public PersonBean[] newArray(int size) {
            return new PersonBean[size];
        }
    };
}

这里就是定义了一个实现 Parcelable 接口的对象,里边定义了 age 和 name 两个参数,其他字段和属性可以自动生成,使用编辑器在当前页面右键选择 generate 然后选择对应的生成内容就会帮我们全部添加进来。

此时可以同步一下工程或者 clean rebuild 一下工程,android studio 会自动生成 IMyAidlInterface.aidl 文件对应的 java 类。

查看此类的代码结构,如图

其中有一个自动生成的代理类 如下所示

public static abstract class Stub extends android.os.Binder implements com.example.myapplication.IMyAidlInterface

可以看到此类就是一个普通的 Binder,并实现了我们写的

IMyAidlInterface 接口。此外还有一个静态方法,

/**
 * Cast an IBinder object into an com.example.myapplication.IMyAidlInterface interface,
 * generating a proxy if needed.
 */
public static com.example.myapplication.IMyAidlInterface asInterface(android.os.IBinder obj)
{
if ((obj==null)) {
return null;
}
android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
if (((iin!=null)&&(iin instanceof com.example.myapplication.IMyAidlInterface))) {
return ((com.example.myapplication.IMyAidlInterface)iin);
}
return new com.example.myapplication.IMyAidlInterface.Stub.Proxy(obj);
}

通过查看这里的源码可以发现,此方法会将服务端的 myService 里返回的IBinder 转换成一个代理类 proxy ,转换过程会判断当前客户端和服务端是否属于同一个进程,如果是则直接返回如果不是则转换成代理类,代理类也在这个 java 文件里。

  1. 然后创建上边提到的服务端的 myService 服务
image

其中的 onBind 方法返回一个 IMyAidlInterface.Stub 类的对象,这里的代码如下

private final IMyAidlInterface.Stub mBinder = new IMyAidlInterface.Stub(){
        @Override
        public void addPerson(PersonBean mBean) throws RemoteException {
           synchronized(mStudents){
               if(!mStudents.contains(mBean)){
                   mStudents.add(mBean);
               }
           }
        }
        @Override
        public List<PersonBean> getPerson() throws RemoteException {
            synchronized (mStudents){
                return mStudents;
            }
        }
        @Override
        public boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException {
            String packageName = null;
            String[] packages = MyService.this.getPackageManager().getPackagesForUid(
                    getCallingUid());
            if(packages!=null&&packages.length>0){
                packageName = packages[0];
            }
            Log.i(TAG,packageName);
            if(!PACKAGE_NAME.equals(packageName)){
                return false;
            }
            return super.onTransact(code, data, reply, flags);
        }
    };
    @Override
    public IBinder onBind(Intent intent) {
        Log.e(TAG, String.format("on bind,intent = %s", intent.toString()));
        displayNotificationMessage("服务已启动");
        return mBinder;
    }

IMyAidlInterface.Stub 类里返回上边 Stub 类里的两个自定义方法,然后将此类的对象 mBinder 在 onBind 里返回供客户端调用。

此外,你可能只想让你的 service 被指定的 apk 应用调用,其他应用不能调用,这时可以重写 Stub 里的 onTransact 方法,并判断当前的 uid 和调用的客户端的 uid 是否一致,如果相同返回 true,否则返回 false ,这样就可以控制特定应用调用的问题。

  1. 在 manifest 里声明当前的 service
<serviceandroid:name=".MyService"
  android:process=":remote"
  android:exported="true">
  <intent-filter>
    <action android:name="com.example.myapplication.MyService"/>      
    <category android:name="android.intent.category.DEFAULT"/>
  </intent-filter>
</service>
  1. 新建一个工程充当客户端,并把服务端里的 aild 包下的所有文件包含 PersonBean.java 文件一起复制到客户端工程里,并且包名不能改变,否则运行的时候会出现 crash 。客户端绑定 service 的代码很简单
package com.example.mysocket11;

import android.app.AlertDialog;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.IBinder;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

import com.example.myapplication.IMyAidlInterface;
import com.example.myapplication.PersonBean;


public class MainActivity extends AppCompatActivity {

   IMyAidlInterface myAidlInterface;
   private final String  ACTION_BIND_SERVICE = "com.example.myapplication.MyService";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        findViewById(R.id.text).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intentService = new Intent(ACTION_BIND_SERVICE);
                intentService.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                intentService.setPackage("com.example.myapplication");
                bindService(intentService,mServiceConnection,BIND_AUTO_CREATE);
            }
        });
    }
    ServiceConnection mServiceConnection = new ServiceConnection(){
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            myAidlInterface = IMyAidlInterface.Stub.asInterface(service);
            try {
                PersonBean toastName = myAidlInterface.getPerson().get(0);
                Toast.makeText(MainActivity.this,toastName.toString(),Toast.LENGTH_SHORT).show();
                showDialog(toastName.toString());
            } catch (Exception e) {
                e.printStackTrace();
                System.out.println(e.toString());
                Log.e("没有获取到数据",e.toString());
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            myAidlInterface = null;
        }
    };
    public void showDialog(String message)
    {
        new AlertDialog.Builder(MainActivity.this)
                .setTitle("an")
                .setMessage(message)
                .setPositiveButton("确定", null)
                .show();
    }
    @Override
    protected void onDestroy() {
        if (myAidlInterface != null) {
            unbindService(mServiceConnection);
        }
        super.onDestroy();
    }
}

先是绑定服务然后在服务连接成功处获取对应的 IBinder 实例,根据这个对象调用服务端的接口里的方法,实现两个不同的进程间获取数据。

最终的效果图如下

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351