AIDL之进程间共享单例

今天是端午节放假第一天,祝各位小伙伴端午快乐!今天我也给大家送个“粽子”-AIDL之进程间共享单例。关于AIDL的简单使用本文不会涉及太多,前面我们有两篇陆续聊了Android AIDL的相关姿势,有需要的小伙伴可以回头看下:
AIDL与Binder浅析//www.greatytc.com/p/4ebd4783d3d9
IPC之Messenger//www.greatytc.com/p/328605278a93

本文会涉及到大概下面这些姿势:

1.进程间通信AIDL的使用步骤
2.AIDL的实现原理
3.单例模式
4.工厂模式
5.模板模式

为什么要写这样一个主题?有时候我们项目中会有这样一个需求,一个应用中有两个进程A和B,进程A中有一些配置放在单例Singleton,进程B需要去拿配置,有哪些方式可以实现呢?一种是序列化的方式,比如写到一个文件,或者写到数据库中;另外一种就是通过Binder的方式,也就是我们今天要说的这种方式。序列化的方式操作会繁琐一点,需要一堆的写入和读出的操作,Binder的方式会更灵活一点。

我们还是通过一个简单的栗子来看下怎么实现的。分成A和B两个进程,A进程和B进程分别有自己的单例Singleton,A进程可以通过B进程的单例调用单例的方法,反过来B进程也可以通过A进程的单例调用单例的方法,操作起来很方便,我们在进程单例中有两个方法和一个成员变量count,方法increment用于递增count,方法getcount用于获取count的值。


1.png
2.png
3.png

1.使用方式

使用起来很简单,先定义两个单例的AIDL文件,因为需要跨进程传输,所以需要AIDL文件。两个单例拥有同样的两个方法,increament中参数是调用方法的当前进程的名称。

interface ISingletonMain {
    void increment(String currentPorcessName);
    int getCount();
}

interface ISingletonB {
    void increment(String currentPorcessName);
    int getCount();
}

当然,为了拿到对方进程的单例,是需要去绑定服务的,在Main进程中绑定B进程的服务:

bindService(ServiceB.class);

B进程绑定Main进程:

bindService(MainService.class);

使用起来就直接使用单例的调用方式就可以,比如调用Main进程单例的方法:

SingletonMainImpl.getInstance().increment(Utils.currentProcessName());

调用B进程单例的方法也是类似:

SingletonBImpl.getInstance().increment(Utils.currentProcessName());

就是这样,使用起来就是一句代码搞定。

2. 原理解析

使用起来很方便,我们来看看背后是怎么实现的。首先应该有个共识,为了共享单例,进程间需要通过Binder通信来拿到对方的实例。为了拿到这个对方的实例,总结起来是这么几个步骤:

1.定义进程间通信的接口。调用这个接口的方法可以拿到对方的单例;
2.绑定服务。A绑定B进程的服务,B进程会生成这个单例,封装成一个Binder,通过Binder池回传一个Binder到A进程;
3.绑定服务成功。成功时,A可以拿到B传送过来的Binder,通过Binder可以取出这个单例;
4.调用单例方法。拿到单例后调用单例接口aidl文件中的方法时会发起IPC调用,在APP层就跟调用本进程单例一样。

我们对应着上面的步骤来看看幕后的实现。

1.定义进程间通信的接口

先来总体看下有多少AIDL接口,先大体介绍下每个接口:

1.IInstanceTransfer.aidl就是Main进程和B进程通信的接口,返回的是进程实例的工厂InstanceFactory
2.实例工厂InstanceFactory为了在AIDL中通过Binder传输也需要定义一个InstanceFactory.aidl文件,并且要实现Parcelable接口;

public class InstanceFactory implements Parcelable

3.剩下两个就是需要传输的进程实例的aidl文件,里面定义了提供跨进程调用的方法:

interface ISingletonMain {
    void increment(String currentPorcessName);
    int getCount();
}
4.png

我们定义一个获取单例的接口,返回的InstanceFactory是一个“工厂”,这又是什么套路?

interface IInstanceTransfer {
    InstanceFactory transfer();
}

我们首先简单讲下工厂模式,比如我现在需要一辆轿车,但是我不可能自己去造一辆车,如果有这么一个车厂Factory,我直接给Factory下个指令“我需要一辆轿车”,Factory就直接造出一辆轿车返回给我,这样是不是很方便,同样另外一个小伙伴B需要SUV,直接给Factory下个指令“我需要一辆SUV”,就能拿到Factory的SUV。
代码直接看我们下面的栗子,我们现在需要Main进程和B进程的单例(至于什么是单例模式我们后面结合代码具体看),那么我们直接给InstanceFactory这个工厂下个指令就行,这样很方便客户的使用。
如果是Main进程,InstanceFactory就造一个Mian进程的单例;如果是B进程,InstanceFactory就造一个B进程的单例。writeStrongInterface方法就是将实例封装成Binder,用于AIDL。

if (Utils.isMainProcess()) {
        dest.writeStrongInterface(SingletonMainImpl.getInstance());
} else if (Utils.isBProcess()) {
        dest.writeStrongInterface(SingletonBImpl.getInstance());
}

2.绑定服务

定义好接口,接着就是绑定服务,APP层其实就是bindService。两个进程都有对应的Service,Main进程对应MainService,B进程对应ServiceB。要实现的东西其实都是一样的,就是返回IInstanceTransfer的实现即可,因此我们将使用另外一种设计模式,模板模式来抽取下代码的实现。我们先拐个弯看下模板模式,讲的会比较简单点,有需要的小伙伴可以去看下设计模式的相关资料。
模板模式其实和继承有很大的关系,就是我们将多个类共性的方法抽取出来放到父类中,每个子类需要个性化实现的就在父类中留个hook给子类自己去实现,但是方法的执行过程其实是在父类的控制中,简单举个栗子,在hookMethod方法的前后可以实现一些其他的工作,我们这里只是简单的打印log。在外面调用TemplateClass的TemplateMethod就可以实现调用同一个模板方法但是有不同实现的效果。

public abstract class TemplateClass {
    public void TemplateMethod(){
        Log.d(TemplateClass.class.toString(), "This is before hookMethod");
        hookMethod();
        Log.d(TemplateClass.class.toString(), "This is after hookMethod");
    }

    public abstract void hookMethod();
}

我们看下Main进程绑定服务的过程,两个进程共同的服务我们抽取出来一个BaseService。

bindService(ServiceB.class);//绑定B进程的服务

public class BaseService extends Service {
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return new InstanceTransferImpl() ;
    }
}

服务返回的就是AIDL接口IInstanceTransfer的实现,这个我们就不展开讲了,有需要的小伙伴自行参考://www.greatytc.com/p/4ebd4783d3d9

public class InstanceTransferImpl extends IInstanceTransfer.Stub {
    @Override
    public InstanceFactory transfer() throws RemoteException {
        return new InstanceFactory();
    }

    @Override
    public boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException {
        try{
            return super.onTransact(code, data, reply, flags);
        }catch (RuntimeException e){
            Log.i("InstanceTransferImpl", "Unexpected exception" + e);
            throw  e;
        }
    }
}

可以看出,调用transfer方法就是返回InstanceFactory。

3.绑定服务成功

绑定服务成功Main进程就能在ServiceConnection中拿到B进程的Binder,其实B进程在绑定Main进程成功时也是要这个步骤,而我们把实例Binder的工作都封装到了InstanceFactory中,因此我们可以用模板模式抽取一个ServiceConnection。

public class ServiceConnectionImpl implements ServiceConnection {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        try {
            InstanceTransferImpl.asInterface(service).transfer();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void onServiceDisconnected(ComponentName name) {
    }
}

这样不管Main进程绑定B进程服务还是B进程绑定Main进程服务都可以使用这个ServiceConnection,还是很方便的。
我们来重点看下绑定服务成功后是怎么拿到单例的。全都浓缩在这句代码里了:

InstanceTransferImpl.asInterface(service).transfer();

跨进程所以我们是拿到InstanceTransferImpl的代理,调用的自然也是代理的transfer,我们分析下这个流程:

1.在代理的transfer()方法中会调用

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

2.来到Stub的onTransact方法中:

com.example.juexingzhe.processshareinstance.InstanceFactory _result = this.transfer();
reply.writeNoException();
if ((_result != null)) {
      reply.writeInt(1);
      _result.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
} else {
     reply.writeInt(0);
}

this当然指的就是B进程,所以调用到B进程的transfer方法,得到InstanceFactor。
3.调用InstanceFactory:
接着会调用 InstanceFactory.writeToParcel,判断是B进程,会在reply中写入信息,我们这边主要进行三个工作,写入B进程标识;写入B进程总的方法调用的统计数字;写入B进程的实例。后面这些信息都可以在Main进程中读出。

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        if (Utils.isMainProcess()) {
            dest.writeInt(PROCESS_MAIN);
            dest.writeInt(SingletonMainImpl.COUNT);
            dest.writeStrongInterface(SingletonMainImpl.getInstance());
        } else if (Utils.isBProcess()) {
            dest.writeInt(PROCESS_B);
            dest.writeInt(SingletonBImpl.COUNT);
            dest.writeStrongInterface(SingletonBImpl.getInstance());
        }
    }

4.再回到代理的transfer方法中,会调用InstanceFactory.CREATOR

_result = com.example.juexingzhe.processshareinstance.InstanceFactory.CREATOR.createFromParcel(_reply);

5.InstanceFactory中读出B进程实例
根据进程标识赋值 SingletonBImpl.INSTANCE

public InstanceFactory createFromParcel(Parcel in) {
            return new InstanceFactory(in);
}
public InstanceFactory(Parcel in) {
        int processId = in.readInt();
        switch (processId) {
            case PROCESS_B:
                SingletonBImpl.COUNT = in.readInt();
                SingletonBImpl.INSTANCE = ISingletonB.Stub.asInterface(in.readStrongBinder());
                break;
        }
}

进过上面过程我们就可以在Main进程拿到B的实例了,但是要注意一点,Main进程拿到的SingletonBImpl.INSTANCE和B进程的实例其实不是真正的同一个Object,因为是跨进程通过Binder池进行传送,但是因为每个Binder都有标识符,所以底层能做出正确的识别。为什么说不是同一个Object,我们可以看个截图。在进程Main中获取单例就是ISingletonMainImpl,这个没什么疑问;但是在进程B中获取的进程Main的单例是ISingletonMainImpl$Stub$Proxy,也就是跨进程的单例代理。从两个实例的mObject也可以看出来不是同一个对象。

5.png
6.png

4.调用单例方法

这个其实就很简单了,就跟这个单例就在Main进程一样。

SingletonBImpl.getInstance().increment(Utils.currentProcessName());
SingletonBImpl.getInstance().getCount()

原理也是通过代理跨进程进行调用,我们以getCount方法的调用为例:

1.通过代理调用到getCount方法:

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

2.调用到B进程单例的getCount方法,在onTransact方法中, 将结果保存到reply中。

 case TRANSACTION_getCount: {
        data.enforceInterface(DESCRIPTOR);
        int _result = this.getCount();
        reply.writeNoException();
        reply.writeInt(_result);
        return true;
}

3.再回到代理的transact方法中,从reply中读出结果并返回。

_result = _reply.readInt();

最后我们看看B进程实例的实现,使用的就是单例模式。关键点就是构造方法是私有的,不允许外面实例化这个单例,只能通过静态方法getInstance获取实例。SingletonBImpl两个方法实现都比较简单没什么可说的,主要是getInstance()方法。主要是B进程那么直接就返回SingletonBImpl,如果是A进程会自动重连。

public class SingletonBImpl extends ISingletonB.Stub {

    public static ISingletonB INSTANCE;

    public static int COUNT = 0;

    private SingletonBImpl() {
    }

    public static ISingletonB getInstance() {

        if (null == INSTANCE) {
            synchronized (SingletonBImpl.class) {
                if (null == INSTANCE) {
                    if (Utils.isBProcess()) {
                        INSTANCE = new SingletonBImpl();
                    } else {
                        Context context = MyApplication.getContext();
                        Intent intent = new Intent(context, ServiceB.class);
                        context.bindService(intent, new ServiceConnectionImpl(), Context.BIND_AUTO_CREATE);
                    }
                }
            }
        }

        return INSTANCE;
    }


    @Override
    public void increment(String currentPorcessName) throws RemoteException {
        COUNT++;
        StringBuilder stringBuilder = new StringBuilder("进程:" + currentPorcessName + "调用" + Utils.currentProcessName() + "的方法increment");
        Utils.showToast(stringBuilder.toString());
    }


    @Override
    public int getCount() {
        return COUNT;
    }
}

2.总结

今天我们这篇文章的AIDL接口会比较多,主要是用到了几个简单的设计模式,可以方便的扩展。主要有进程通信的接口,还有“工厂”的接口,另外两个就是进程单例的接口。为了实现两个进程共享单例,我们是通过AIDL接口IInstanceTransfer获取一个“工厂”,工厂会自动根据当前的进程实例化进程实例,通过Binder传输这个单例,在跨进程拿到这个单例后就可以调用进程单例AIDL接口定义的方法来实现各种操作。要注意的一点就是拿到单例后也是通过跨进程AIDL进行类单例调用的,这就是为什么文章前面需要定义两个进程单例AIDL接口的原因。还是那句话,建议结合前面AIDL入门的文章看会比较好点。
文章链接://www.greatytc.com/p/4ebd4783d3d9
本文栗子也已经上传Github:https://github.com/juexingzhe/ProcessShareInstance

完结,谢谢!

参考链接:
//www.greatytc.com/p/4ebd4783d3d9
https://toutiao.io/posts/6yvvo2/preview

欢迎关注公众号:JueCode

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

推荐阅读更多精彩内容