AIDL設計思路

AIDL設計思路

AIDL

1.實現一個自定義的AIDL

// IMyService.aidl
package com.yetao.testaidl;

interface IMyService {
    /**
     * Demonstrates some basic types that you can use as parameters
     * and return values in AIDL.
     */
    void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
            double aDouble, String aString);
}

上面聲明瞭aidl接口、帶基礎數據類型的方法

2.IDE自動生成IMyService.java文件

    /*
 * This file is auto-generated.  DO NOT MODIFY.
 */
package com.yetao.testaidl;
// Declare any non-default types here with import statements

public interface IMyService extends android.os.IInterface {
    /**
     * IMyService的默認實現
     */
    public static class Default implements com.yetao.testaidl.IMyService {
        /**
         * Demonstrates some basic types that you can use as parameters
         * and return values in AIDL.
         */
        @Override
        public int basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) throws android.os.RemoteException {
            return 0;
        }

        @Override
        public android.os.IBinder asBinder() {
            return null;
        }
    }

    /**
     * 本地IPC需要實現子類
     */
    public static abstract class Stub extends android.os.Binder implements com.yetao.testaidl.IMyService {
        private static final java.lang.String DESCRIPTOR = "com.yetao.testaidl.IMyService";

        /**
         * 構造函數,將標識符賦給當前對象
         */
        public Stub() {
            this.attachInterface(this, DESCRIPTOR);
        }

        /**
         *將IBinder對象轉換爲com.yetao.testaidl.IMyService接口,並在需要時生成代理。
        */
        public static com.yetao.testaidl.IMyService asInterface(android.os.IBinder obj) {
            if ((obj == null)) {
                return null;
            }
            //通過DESCRIPTOR查找本地是否有對應接口
            android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
            //如果本地找到了,就返回
            if (((iin != null) && (iin instanceof com.yetao.testaidl.IMyService))) {
                return ((com.yetao.testaidl.IMyService) iin);
            }
            //沒有的話,就構建遠程代理對象
            return new com.yetao.testaidl.IMyService.Stub.Proxy(obj);
        }

        @Override
        public android.os.IBinder asBinder() {
            return this;
        }

        @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 INTERFACE_TRANSACTION: {
                    reply.writeString(descriptor);
                    return true;
                }
                case TRANSACTION_basicTypes: {
                    data.enforceInterface(descriptor);
                    int _arg0;
                    _arg0 = data.readInt();
                    long _arg1;
                    _arg1 = data.readLong();
                    boolean _arg2;
                    _arg2 = (0 != data.readInt());
                    float _arg3;
                    _arg3 = data.readFloat();
                    double _arg4;
                    _arg4 = data.readDouble();
                    java.lang.String _arg5;
                    _arg5 = data.readString();
                    int _result = this.basicTypes(_arg0, _arg1, _arg2, _arg3, _arg4, _arg5);
                    reply.writeNoException();
                    reply.writeInt(_result);
                    return true;
                }
                default: {
                    return super.onTransact(code, data, reply, flags);
                }
            }
        }

        private static class Proxy implements com.yetao.testaidl.IMyService {
            private android.os.IBinder mRemote;

            Proxy(android.os.IBinder remote) {
                mRemote = remote;
            }

            @Override
            public android.os.IBinder asBinder() {
                return mRemote;
            }

            public java.lang.String getInterfaceDescriptor() {
                return DESCRIPTOR;
            }

            /**
             * 演示一些基本類型,您可以將它們用作參數並在AIDL中返回值。
             */
            @Override
            public int basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) 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.writeInt(anInt);
                    _data.writeLong(aLong);
                    _data.writeInt(((aBoolean) ? (1) : (0)));
                    _data.writeFloat(aFloat);
                    _data.writeDouble(aDouble);
                    _data.writeString(aString);
                    boolean _status = mRemote.transact(Stub.TRANSACTION_basicTypes, _data, _reply, 0);
                    if (!_status && getDefaultImpl() != null) {
                        return getDefaultImpl().basicTypes(anInt, aLong, aBoolean, aFloat, aDouble, aString);
                    }
                    _reply.readException();
                    _result = _reply.readInt();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
                return _result;
            }

            public static com.yetao.testaidl.IMyService sDefaultImpl;
        }

        //方法對應Id,依次增加
        static final int TRANSACTION_basicTypes = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);

        public static boolean setDefaultImpl(com.yetao.testaidl.IMyService impl) {
            if (Stub.Proxy.sDefaultImpl == null && impl != null) {
                Stub.Proxy.sDefaultImpl = impl;
                return true;
            }
            return false;
        }

        public static com.yetao.testaidl.IMyService getDefaultImpl() {
            return Stub.Proxy.sDefaultImpl;
        }
    }

    /**
     * 演示一些基本類型,您可以將它們用作參數並在AIDL中返回值。
     */
    public int basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) throws android.os.RemoteException;

}

自動生成的java類,主要有以下幾個功能:
1.生成接口類IMyService(實現了IInterface接口來轉換成Binder對象)
2.生成一個默認實現類Default(IMyService的實現類)
3.生成本地IPC的實現類-Stub(Binder子類,實現IMyService接口),用於本地實現對應的Binder服務,避免不必要的遠程IPC通信,同時也能兼顧本地提供服務
4.生成遠端IPC的實現類-Proxy(實現IMyService接口),通過傳入的遠程Binder對象,調用transact()和遠端通信,然後獲取reply返回值

3.本地IPC-Stub本地的使用

public class AidlService extends Service {

    private IMyService.Stub stub = new IMyService.Stub() {
        @Override
        public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException {

        }
    };


    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return stub;
    }
}


上面主要做了:
1.繼承IMyService.Stub,實現basicTypes方法
2.將stub對象賦值給Service的onBind()綁定,這也是爲什麼IMyService.Stub要繼承Binder的原因,需要來兼容本地IPC

4.遠端IPC使用

public class ClientActivity extends AppCompatActivity {

    private IMyService aidlService;

    private boolean connected;

    private void bindMyService() {
        ServiceConnection serviceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                aidlService = IMyService.Stub.asInterface(service);
                connected = true;
            }

            @Override
            public void onServiceDisconnected(ComponentName name) {
                connected = false;
            }
        };
        //遠端訪問應該使用隱式啓動,這裏爲了方便,使用顯式調用
        Intent intent = new Intent(this, AidlService.class);
        bindService(intent,serviceConnection, Context.BIND_AUTO_CREATE);
    }

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        bindMyService();
        initListener();
    }

    private void initListener() {
        findViewById(R.id.fab).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if(connected){
                    try {
                        aidlService.basicTypes(1,2,false,0f,0,"sa");
                    } catch (RemoteException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
    }
}

使用第三部的AidlService,上面做了幾步:
1.構建連接回調ServiceConnection,監聽onServiceConnected()方法,獲取到遠端的binder對象,轉換爲IMyService類型對象
2.傳入ServiceConnection啓動服務,供回調
3.點擊按鈕,通過獲取到的binder對象(IMyService)調用遠端方法做交互

5.aidl整體流程

從basicTypes() -> transact() -> onTransact() 來進行分析

上面Proxy中的basicTypes()裏面調用Binder對象的transact()方法

Binder.java

    /**
     * 默認實現倒回包裹並在onTransact上調用。在遠端,將調用交易到活頁夾中以進行IPC。
     */
    public final boolean transact(int code, @NonNull Parcel data, @Nullable Parcel reply,
            int flags) throws RemoteException {
        if (false) Log.v("Binder", "Transact: " + code + " to " + this);

        if (data != null) {
            data.setDataPosition(0);
        }
        boolean r = onTransact(code, data, reply, flags);
        if (reply != null) {
            reply.setDataPosition(0);
        }
        return r;
    }

1.調用服務端的Binder對象實現的onTransact方法
2.設置數據位置爲0

來看Service中繼承的Stub的類

public static abstract class Stub extends android.os.Binder implements com.yetao.testaidl.IMyService {
        private static final java.lang.String DESCRIPTOR = "com.yetao.testaidl.IMyService";
        ``````省略無關代碼

        @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 INTERFACE_TRANSACTION: {
                    reply.writeString(descriptor);
                    return true;
                }
                case TRANSACTION_basicTypes: {
                    data.enforceInterface(descriptor);
                    int _arg0;
                    _arg0 = data.readInt();
                    long _arg1;
                    _arg1 = data.readLong();
                    boolean _arg2;
                    _arg2 = (0 != data.readInt());
                    float _arg3;
                    _arg3 = data.readFloat();
                    double _arg4;
                    _arg4 = data.readDouble();
                    java.lang.String _arg5;
                    _arg5 = data.readString();
                    int _result = this.basicTypes(_arg0, _arg1, _arg2, _arg3, _arg4, _arg5);
                    reply.writeNoException();
                    reply.writeInt(_result);
                    return true;
                }
                default: {
                    return super.onTransact(code, data, reply, flags);
                }
            }
        }
        `````
}


主要步驟:
1.客戶端調用transact()發送TRANSACTION_basicTypes(code)後,服務端onTransact()接受到
2.從data讀出客戶端傳過來的數據
3.調用對應實現方法(basicTypes)獲取到返回值,寫入到reply;如果沒有返回值,則不寫入

小結,截止目前,瞭解到了在應用層aidl使用所做的流程,將在下一篇說明具體Binder驅動是怎麼交互的

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