关于 AIDL 的一些注意事项

AIDL

  1. 服务端创建一个 Service 监听客户端的链接请求,将 AIDL 的实现回调给客户端;

    客户端通过 aidl 就可以直接调用服务端的方法

  2. AIDL 的声明注意点:

    • C/S 两端必须完全一致,包名都不能错,否则找不到
    • AIDL 中支持的数据类型不多:
      • 基本数据类型、String、char
      • List 只支持 ArrayList,Map 只支持 HashMap,且每个元素都必须被 AIDL 所支持
        • AIDL 支持的其实是抽象的 List 和 Map,并在最终返回 ArrayList 和 HashMap
        • 在 Server 方法中,可以使用其他数据类型,比如 CopyOnWriteArrayList 和 ConcurrentHashMap
      • parcelable 对象(必须显示 import)
        • 必须声明一个同样的 aidl 文件,声明其为 parcelable 对象
      • AIDL 接口本身(必须显示 import)
    • 除基本类型外,所有入参必须标明 in / out / inout
      • in 代表输入型参数,out 代表输出型参数,inout 表示输入输出型参数
    • 只支持方法,不支持静态常量
    • AIDL 的方法是在 Binder 线程池中执行的,所以一般需要处理线程同步问题
      • 使用 CopyOnWriteArrayList、ConcurrentHashMap、AtomicBoolean
    • 如果有接口回调,从 binder 回调到 app
      • 使用 aidl 而不是普通接口
      • 注册、解注册需要使用 RemoteCallbackListener,而不是常规方式
        • 注意 RemoteCallbackListener 的使用,begin & finish 成对使用
    • 可以使用权限或者包名的方式,在 onTransact 或者 onBind 方法中校验连接者

AIDL 代码

// IBookManager.aidl
package com.test.testaidl_1;
import com.test.testaidl_1.Book;
import com.test.testaidl_1.IOnNewBookArrivedListener;
interface IBookManager {
    List<Book> getBookList();
    void addBook(in Book book);
    void registerListener(IOnNewBookArrivedListener l);
    void unregisterListener(IOnNewBookArrivedListener l);
}
// Book.aidl.aidl
package com.test.testaidl_1;
parcelable Book;

服务端代码

public class BookManagerService extends Service {

    private static final String TAG = "bms";

    // 线程安全的布尔对象
    private AtomicBoolean mIsServiceDestroyed = new AtomicBoolean(false);

    // 线程安全的 List
    private CopyOnWriteArrayList<Book> mBookList = new CopyOnWriteArrayList<>();

    private RemoteCallbackList<IOnNewBookArrivedListener> mListenerList
            = new RemoteCallbackList<>();

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {

        int check = checkCallingOrSelfPermission("ACCESS_BALABALA");
        if (check == PackageManager.PERMISSION_DENIED) {
            Log.e(TAG, "permission denied");
            return null;
        } else {
            Log.e(TAG, "permission granted");
        }

        return mBinder;
    }

    private Binder mBinder = new IBookManager.Stub() {
        @Override
        public List<Book> getBookList() throws RemoteException {
//            SystemClock.sleep(5000);
            return mBookList;
        }

        @Override
        public void addBook(Book book) throws RemoteException {
            mBookList.add(book);
        }

        @Override
        public void registerListener(IOnNewBookArrivedListener l) throws RemoteException {
            mListenerList.register(l);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
                Log.e(TAG, "register list size " + mListenerList.getRegisteredCallbackCount());
            }
        }

        @Override
        public void unregisterListener(IOnNewBookArrivedListener l) throws RemoteException {
            mListenerList.unregister(l);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
                Log.e(TAG, "register list size " + mListenerList.getRegisteredCallbackCount());
            }
        }
    };

    @Override
    public void onCreate() {
        super.onCreate();
        new Thread(new ServiceWorker()).start();
    }

    private class ServiceWorker implements Runnable {
        @Override
        public void run() {
            while (!mIsServiceDestroyed.get()) {
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                int bookId = mBookList.size() + 1;
                Book newBook = new Book(bookId, "new Book#" + bookId);
                onNewBookArrived(newBook);
            }
        }
    }

    private void onNewBookArrived(Book newBook) {
             // 运行在 binder 线程池中
        SystemClock.sleep(6000);

        mBookList.add(newBook);
        int size = mListenerList.beginBroadcast();
        for (int i = 0; i < size; i++) {
            try {
                IOnNewBookArrivedListener broadcastItem = mListenerList.getBroadcastItem(i);
                if (broadcastItem != null) {
                     // 客户端回调,如果更新 UI,需要使用 handler 切换线程,否则会报错
                    broadcastItem.onNewBookArrived(newBook);
                }
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        mListenerList.finishBroadcast();
    }

}

客户端代码

public class MainActivity extends AppCompatActivity {

    public static final String TAG = "MainActivity";

    private IBookManager mBookManager;

    private MyListener myListener = new MyListener();

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


        Intent intent = new Intent(this, BookManagerService.class);
        bindService(intent, mConn, Context.BIND_AUTO_CREATE);

    }

    private ServiceConnection mConn = new ServiceConnection() {

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            mBookManager = IBookManager.Stub.asInterface(service);

            try {
                List<Book> bookList = mBookManager.getBookList();
                Log.e(TAG, "query book list, type is " + bookList.getClass().getCanonicalName());
                Log.e(TAG, "query book list: " + Arrays.toString(bookList.toArray()));

                mBookManager.addBook(new Book(3, "这是本新书"));
                Log.e(TAG, "new book");
                bookList = mBookManager.getBookList();
                Log.e(TAG, "query book list: " + Arrays.toString(bookList.toArray()));

                // 这是 UI 线程
                mBookManager.registerListener(myListener);

            } catch (RemoteException e) {
                e.printStackTrace();
            }

        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }

    };

    public void testBinder(View view) {

        Toast.makeText(this,"999",Toast.LENGTH_SHORT).show();

         //  mBookManager.getBookList(); 模拟耗时操作
        // 若不放在子线程中,就会 ANR
        new Thread(){
            @Override
            public void run() {
                super.run();

                try {
                    List<Book> bookList = mBookManager.getBookList();
                    Log.e(TAG, "query book list: " + Arrays.toString(bookList.toArray()));

                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
        }.start();

    }

    private class MyListener extends IOnNewBookArrivedListener.Stub {
        @Override
        public void onNewBookArrived(Book book) throws RemoteException {
            // 此方法运行在 Binder 线程的线程池中
            // 如果在这里做操作,就会抛异常,不是在 Looper 线程
            Toast.makeText(MainActivity.this,
                    "666",Toast.LENGTH_SHORT).show();
            mHandler.obtainMessage(-1, book).sendToTarget();
        }
    }

    private Handler mHandler = new Handler(){
        @Override
        public void handleMessage(Message msg) {
            if (msg.what == -1) {
                Log.e(TAG, "收到新书:" + msg.obj.toString());
            }
        }
    };

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (mBookManager != null && mBookManager.asBinder().isBinderAlive()) {
            try {
                mBookManager.unregisterListener(myListener);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        unbindService(mConn);
    }
}

以这个 AIDL 实现为例,简单分析一下代码走向。

  1. 客户端发起绑定远程服务的请求,经过源码走向,会走向指定的服务信息;

  2. 服务或者为本地服务,或者为其他应用(远端服务),都会通过 onBind 方法将实例化好的 Binder 对象返回给客户端;

    // Stub extends Binder
    private Binder mBinder = new IBookManager.Stub() {}
    
  3. 然后代码就会走到客户端的 onServiceConnected 回调:

    private ServiceConnection mConn = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
          mBookManager = IBookManager.Stub.asInterface(service);
        }
    }
    

    具体我们来看一下系统自动生成的 IBookManager 的 java 类:

    public interface IBookManager extends android.os.IInterface {
    
        public static abstract class Stub extends Binder implements IBookManager {
    
            public Stub() {
                this.attachInterface(this, DESCRIPTOR);
            }
    
            public static IBookManager asInterface(android.os.IBinder obj) {
                if ((obj == null)) {
                    return null;
                }
                android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
                if (((iin != null) && (iin instanceof IBookManager))) {
                    return ((IBookManager) iin);
                }
                return new IBookManager.Stub.Proxy(obj);
            }
    
            @Override
            public boolean onTransact(... ...) {
                switch (code) {
                    case TRANSACTION_getBookList: {
                        data.enforceInterface(DESCRIPTOR);
                        java.util.List<com.ljt.testaidl_1.Book> _result = this.getBookList();
                        reply.writeNoException();
                        reply.writeTypedList(_result);
                        return true;
                    }
                }
                return super.onTransact(code, data, reply, flags);
            }
    
            private static class Proxy implements com.ljt.testaidl_1.IBookManager {
                private android.os.IBinder mRemote;
    
                Proxy(android.os.IBinder remote) {
                    mRemote = remote;
                }
    
                @Override
                public java.util.List<com.ljt.testaidl_1.Book> getBookList()  {
                    Parcel _data = Parcel.obtain();
                    Parcel _reply = Parcel.obtain();
                    java.util.List<com.ljt.testaidl_1.Book> _result;
                    try {
                        _data.writeInterfaceToken(DESCRIPTOR);
                        mRemote.transact(Stub.TRANSACTION_getBookList, _data, _reply, 0);
                        _reply.readException();
                        _result = _reply.createTypedArrayList(com.ljt.testaidl_1.Book.CREATOR);
                    } finally {
                        _reply.recycle();
                        _data.recycle();
                    }
                    return _result;
                }
               
            }
    
            static final int TRANSACTION_getBookList = ... ...;
        }
    
        public java.util.List<com.ljt.testaidl_1.Book> getBookList();
    
    }
    
    

    Stub 类实际上就是个 Binder,用来进行序列化、反序列化操作;Proxy 是服务端在客户端的远程代理,当服务端不在本地时,才会使用到。

  4. 如果服务在本地,那在 service connect 的时候,Binder 就会找到 aidl 对应的本地实现,并将该对象返回回去;

    如果服务不在本地,就会使用代理对象:

    public static IBookManager asInterface(android.os.IBinder obj) {
        android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
        if (((iin != null) && (iin instanceof IBookManager))) {
          return ((IBookManager) iin);
        }
        return new IBookManager.Stub.Proxy(obj);
    }
    
  5. 如果服务在本地,那之后的交互就很简单了,因为对应的实现类已经找到,直接用引用进行调用即可,涉及不到 binder 通信;

  6. 如果服务在远端,当用户发起方法调用时,会进入到 Proxy 的方法中:

    public java.util.List<com.ljt.testaidl_1.Book> getBookList()  {
        Parcel _data = Parcel.obtain();
        Parcel _reply = Parcel.obtain();
        java.util.List<com.ljt.testaidl_1.Book> _result;
        try {
          _data.writeInterfaceToken(DESCRIPTOR);
          mRemote.transact(Stub.TRANSACTION_getBookList, _data, _reply, 0);
          _reply.readException();
          _result = _reply.createTypedArrayList(com.ljt.testaidl_1.Book.CREATOR);
        } finally {
          _reply.recycle();
          _data.recycle();
        }
        return _result;
    }
    

    根据代码来看,当客户端发起调用时,会先获取两个 Parcel 对象,并且通过 transact 方法传递给服务端。

    同时,客户端线程挂起,等待服务端返回数据;

    public final boolean transact(int code, Parcel data, Parcel reply,
                                  int flags) throws RemoteException {
      boolean r = onTransact(code, data, reply, flags);
      return r;
    }
    

    根据 Binder 源码可知,会回调到 onTransact 方法,也就是 Proxy 的 onTransact 方法:

    @Override
    public boolean onTransact(... ...) {
      switch (code) {
        case TRANSACTION_getBookList: {
          data.enforceInterface(DESCRIPTOR);
          java.util.List<com.ljt.testaidl_1.Book> _result = this.getBookList();
          reply.writeNoException();
          reply.writeTypedList(_result);
          return true;
        }
      }
      return super.onTransact(code, data, reply, flags);
    }
    

    经过 onTransact 方法的处理,会像 reply 的 parcel 对象中写入返回值,之后该方法返回 true 唤醒客户端,继续执行,从而完成整个调用。

以上,大概就是使用 AIDL 进行 binder 通信的过程。

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

推荐阅读更多精彩内容