美文网首页Android开发专栏Android开发经验谈Android开发
Android 的进程间通信 Binder——AIDL的入门使用

Android 的进程间通信 Binder——AIDL的入门使用

作者: 在代码下成长 | 来源:发表于2017-12-02 21:25 被阅读46次

    进程间通信系列

    AIDL的入门使用(一)
    AIDL的入门使用(二)
    AIDL的入门使用(三)
    Messenger的入门使用

    序言:

    1、AIDL的大致使用流程:首先是创建一个AIDL接口文件声明需要在客户端调用的接口,再创建一个Service,接着创建一个类继承自AIDL接口中的Stub类并实现Stub中的抽象方法,在Service 的onBind方法中返回这个类的对象,然后在客户端就可以绑定服务端的Service,建立连接后就可以访问远程服务端的方法了;
    2、思考:当有多个AIDL接口文件时怎么办呢?按照以前的思路,只有创建多个Service了,或者将多个AIDL接口的方法放在一个AIDL中了,这回造成代码耦合程度高,这篇我们主要就是解决这个问题的。解决思路:在服务端统一返回一个AIDL文件的binder,这个binder对象只有一个query方法,可以根据传入的值去创建其他对应的AIDL接口的Binder对象在进行返回,这样客户端就可以单独调用服务端的AIDL接口中的方法了,并且只需要一个Service。

    多个AIDL接口通信服务端:

    1、新建三个AIDL接口文件,分别为:IStringComputface、IIntComputInterface 和IBinderInterface ,前两个提供给客户端具体的调用方法,IBinderInterface 提供给客户端查询调用的Binder对象。然后Make Project。

    // IStringComputface.aidl
    package com.ljp.moreaidl_server;
    //计算字符串的AIDL接口
    interface IStringComputInterface {
        String comput(String str);
    }
    
    // IBinderInterface.aidl
    package com.ljp.moreaidl_server;
    interface IBinderInterface {
        IBinder quary(int binderCode);//根据binder码查询Binder对象
    }
    
    // IBinderInterface.aidl
    package com.ljp.moreaidl_server;
    interface IBinderInterface {
        IBinder quary(int binderCode);//根据binder码查询Binder对象
    }
    

    2、创建3个类分别继承对应AIDL接口的Stub类并实现未实现的方法。

    package com.ljp.moreaidl_server;
    import android.os.RemoteException;
    import android.text.TextUtils;
    public class StringComputImp extends IStringComputInterface.Stub {
        @Override
        public String comput(String str) throws RemoteException {
            if (TextUtils.isEmpty(str)) {
                return null;
            }
            char[] charArray = str.toCharArray();
            for (int i = 0; i < charArray.length; i++) {
                charArray[i] += 1;
            }
            return new String(charArray);
        }
    }
    
    package com.ljp.moreaidl_server;
    import android.os.RemoteException;
    public class IntComputImp extends IIntComputInterface.Stub {
        @Override
        public int addComput(int x, int y) throws RemoteException {
            return x + y;
        }
    }
    
    package com.ljp.moreaidl_server;
    import android.os.IBinder;
    import android.os.RemoteException;
    public class BinderImp extends IBinderInterface.Stub {
        public static final int ComputString = 0;
        public static final int ComputInt = 1;
        @Override
        public IBinder quary(int binderCode) throws RemoteException {
            IBinder binder = null;
            switch (binderCode) {
                case ComputString:
                    binder = new StringComputImp();
                    break;
                case ComputInt:
                    binder = new IntComputImp();
                    break;
            }
            return binder;
        }
    }
    

    3、创建一个Service,并在onBinder方法中返回IBinderInterface的实现类BinderImp的对象,并在注册Service。服务端的代码就完成了。

    package com.ljp.moreaidl_server;
    import android.app.Service;
    import android.content.Intent;
    import android.os.IBinder;
    public class MyServerService extends Service {
        BinderImp mBinderImp = new BinderImp();
        @Override
        public IBinder onBind(Intent intent) {
            return mBinderImp;
        }
    }
    

    在AndroidMainfest文件中注册Service。

            <service
                android:name=".MyServerService"
                android:enabled="true"
                android:exported="true">
                <!-- 程序的包名:com.ljp.moreaidl_server 绑定服务端时使用服务端程序的包名。-->
                <intent-filter>
                    <action android:name="com.ljp.moreAidl.server.action"/>
                </intent-filter>
            </service>
    

    多个AIDL接口通信客户端:

    1、这里写一个Binder连接池类,采用单例模式,在创建对象时就去绑定服务端,绑定成功后会回调ServiceConnection对象的onServiceConnected方法,在onServiceConnected方法中给binder设置死亡代理,当服务端的binder对象死亡时系统回调mDeathRecipient.binderDied()方法从而进行重连。并暴露一个quaryBinder的方法,

    package com.ljp.moreaidl_client;
    import android.content.ComponentName;
    import android.content.Context;
    import android.content.Intent;
    import android.content.ServiceConnection;
    import android.os.IBinder;
    import android.os.RemoteException;
    import android.util.Log;
    import com.ljp.moreaidl_server.IBinderInterface;
    public class BinderPool {
        private static final String TAG = "moreAidl";
        public static final int ComputString = 0;
        public static final int ComputInt = 1;
        private static volatile BinderPool instance;
        private Context mContext;
        private BinderPool(Context context) {
            mContext = context.getApplicationContext();
            connectService();
        }
        public static BinderPool getInstance(Context context) {
            if (instance == null) {
                synchronized (BinderPool.class) {
                    if (instance == null) {
                        instance = new BinderPool(context);
                    }
                }
            }
            return instance;
        }
        IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {
            @Override
            public void binderDied() {
                //当Binder死亡时,系统会回调该方法,在此移除之前绑定的Binder代理并重新绑定远程服务
                Log.e(TAG, "binderDied: 与服务端连接的Binder对象死亡。请求重连!");
                mIBinderInterface.asBinder().unlinkToDeath(mDeathRecipient, 0);
                mIBinderInterface = null;
                connectService();
            }
        };
        IBinderInterface mIBinderInterface = null;
        ServiceConnection mConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                mIBinderInterface = IBinderInterface.Stub.asInterface(service);//返回一个可以查询其他AIDL接口的Binder。
                try {
                    //给binder设置死亡代理,当服务端的binder对象死亡时系统回调mDeathRecipient.binderDied()方法
                    mIBinderInterface.asBinder().linkToDeath(mDeathRecipient, 0);
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
                Log.e(TAG, "onServiceConnected: 与服务端连接成功");
            }
            @Override
            public void onServiceDisconnected(ComponentName name) {
                Log.e(TAG, "onServiceDisconnected: 与服务端断开连接");
                mDeathRecipient.binderDied();
            }
        };
        private synchronized void connectService() {
            Intent intent = new Intent();
            intent.setPackage("com.ljp.moreaidl_server");
            intent.setAction("com.ljp.moreAidl.server.action");
            boolean success = mContext.bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
            Log.e(TAG, "connectService: 绑定服务端=" + success);
        }
        public synchronized void unConnect() { //断开与服务端的连接
            if (mIBinderInterface != null) {
                mContext.unbindService(mConnection);//不会回调onServiceDisconnected()
                mIBinderInterface.asBinder().unlinkToDeath(mDeathRecipient, 0);
            }
            mIBinderInterface = null;
            instance = null;
        }
        public IBinder quaryBinder( int binderCode) { //AIDL的调用都是耗时操作,建议放在子线程中
            IBinder binder = null;
            if (mIBinderInterface != null) {
                try {
                    binder = mIBinderInterface.quary(binderCode); //调用服务端的查询
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
            return binder;
        }
    }
    

    2、在客户端中调用服务端的方法时,先调用BinderPool对象的quaryBinder( int ),然后在使用对应的AIDL接口.Stub.asInterface(binder)进行转换为相应的AIDL接口对象,这时就可以调用该AIDL接口服务端的方法了,由于AIDL的调用都是耗时操作,建议都放在子线程中处理。这里第一次绑定时并不会立即连接成功,需要等待一小段时间,因此建议在Application中获取一次BinderPool的对象进行绑定连接,后面就可以直接使用了。

        public void StringComput(View view) {
            IBinder binder = BinderPool.getInstance(this).quaryBinder(BinderPool.ComputString);
            if (binder == null) {
                return;
            }
            IStringComputInterface anInterface = IStringComputInterface.Stub.asInterface(binder);
            try {
                String comput = anInterface.comput("abcdefg"); //AIDL的调用都是耗时操作,建议放在子线程中
                Log.e(TAG, "StringComput: comput=" + comput);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        public void IntComput(View view) {
            IBinder binder = BinderPool.getInstance(this).quaryBinder(BinderPool.ComputInt);
            if (binder == null) {
                return;
            }
            IIntComputInterface anInterface = IIntComputInterface.Stub.asInterface(binder);
            try {
                int addComput = anInterface.addComput(10, 20);//AIDL的调用都是耗时操作,建议放在子线程中
                Log.e(TAG, "StringComput: addComput=" + addComput);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        @Override
        protected void onDestroy() {
            BinderPool.getInstance(this).unConnect();
            super.onDestroy();
        }
    

    测试结果:

    12-02 21:16:04.176 27386-27386/com.ljp.moreaidl_client E/moreAidl: connectService: 绑定服务端=true
    12-02 21:16:04.186 27386-27386/com.ljp.moreaidl_client E/moreAidl: onServiceConnected: 与服务端连接成功
    12-02 21:16:05.976 27386-27386/com.ljp.moreaidl_client E/moreAidl: StringComput: comput=bcdefgh
    12-02 21:16:07.536 27386-27386/com.ljp.moreaidl_client E/moreAidl: StringComput: addComput=30
    

    更多参考:

    我的CSDN博客地址:http://blog.csdn.net/wo_ha/article/details/78698458

    相关文章

      网友评论

        本文标题:Android 的进程间通信 Binder——AIDL的入门使用

        本文链接:https://www.haomeiwen.com/subject/dwrobxtx.html