Android中AIDL的使用详解,如何发起回调?

这是百度面试官问的一个问题,当时没答上来。我们知道AIDL底层是基于Binder机制通信的,而Binder本身是C/S架构的。Activity写个AIDL接口可以实现跟Service的通信,那么Service如何主动回调或者主动推送消息到Activity呢?

定义通信接口

这个接口是Activity发数据给Service用的,addPerson会在Service中的List中新增一个数据,getPersonList返回Person列表。registerCallback注册回调对象,等一会儿会说。

package com.billshen.offerlearn.service;
import com.billshen.offerlearn.service.IPersonCallBack;
interface IPerson {
    void addPerson(String name);
    List<String> getPersonList();
    void registerCallback(IPersonCallBack cb);
    void unregisterCallback(IPersonCallBack cb);
}

IPersonCallBack.aidl这个接口是Service主动回调Activity用的,用于获取Activity中的时间。

package com.billshen.offerlearn.service;
interface IPersonCallBack {
    String getTime();
}

在AndroidManifest中定义service为另一个进程,并指明action

<service
    android:name="com.billshen.offerlearn.service.AIDLService"
    android:enabled="true"
    android:exported="true"
    android:process=":aidl_service">
    <intent-filter>
        <action android:name="com.aidl.person" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</service>

定义完成后,在AS中build一下,编译器会自动为我们生成binder抽象对象,当然你也可以自己写。

IPerson是个interface继承于IInterface

 

内部包含我刚才在AIDL里面声明的四个方法。

 

还有三个内部静态类Default,Stub和Proxy。其中Stub交给服务方实现,Proxy交给客户端持有。Proxy主要是怎么往内核空间写数据和拿数据,Default是默认实现,基本不用。

 

 

 

 

 

实现具体方法

编译器为我们生成的IPerson.java文件中有个Stub()内部静态类,这是个抽象类,具体实现需要我们自己去写。RemoteCallbackList是一个回调对象列表,可以把Service想象成一个服务器,多个Activity为客户端。Service在建立连接的时候保存了客户端的回调对象。

public class AIDLService extends Service {
    private String TAG = AIDLService.class.getSimpleName();
    private RemoteCallbackList<IPersonCallBack> iPersonCallBacks = new RemoteCallbackList<>();//回调对象列表
    List<String> mPersons = new ArrayList<>();

    public AIDLService() {
        mPersons.add("初始人");
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        Log.i(TAG, "onBind: " + mPersons);
        BroadcastThread broadcastThread = new BroadcastThread();
        broadcastThread.start();
        return new IPerson.Stub() {
            @Override
            public void addPerson(String name) throws RemoteException {
                mPersons.add(name);
            }

            @Override
            public List<String> getPersonList() throws RemoteException {
                return mPersons;
            }

            @Override
            public void registerCallback(IPersonCallBack cb) throws RemoteException {
                iPersonCallBacks.register(cb);//注册回调对象
            }

            @Override
            public void unregisterCallback(IPersonCallBack cb) throws RemoteException {
                iPersonCallBacks.unregister(cb);//反注册回调对象
            }
        };
    }

    class BroadcastThread extends Thread {
        @Override
        public void run() {
            while (true) {
                try {
                    Thread.sleep(1000);
                    int len = iPersonCallBacks.beginBroadcast();//开始发送广播
                    for (int i = 0; i < len; i++) {//遍历回调对象,将拿回来的数据打印再控制台上
                        Log.i(TAG, "run: " + iPersonCallBacks.getBroadcastItem(0).getTime());
                    }
                    iPersonCallBacks.finishBroadcast();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

 Activity中实现客户端的回调函数和开启新线程不断让Service增加Person并拿回数据显示在TextView上。

public class AIDLActivity extends BaseMvpActivity {

    IPerson iPerson = null;
    TextView resultTv;
    Handler mHandler = new Handler() {
        @Override
        public void handleMessage(@NonNull Message msg) {
            try {
                resultTv.setText(iPerson.getPersonList().toString());
            } catch (RemoteException e) {
                e.printStackTrace();
            }
            super.handleMessage(msg);
        }
    };

    @Override
    public void initView() {
        Log.i(TAG, "initView: ");
        resultTv = findViewById(R.id.result_tv);
    }

    @Override
    public void initData() {
        Log.i(TAG, "initData: ");
        Intent intent = new Intent();
        intent.setPackage("com.billshen.offerlearn");
        intent.setAction("com.aidl.person");
        ServiceConnection serviceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                iPerson = IPerson.Stub.asInterface(service);
                IPersonCallBack iPersonCallBack = new IPersonCallBack.Stub() {
                    @Override
                    public String getTime() throws RemoteException {
                        //回调函数,返回时间给服务端
                        SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                        return df.format(System.currentTimeMillis());
                    }
                };
                try {
                    iPerson.registerCallback(iPersonCallBack);//在Service中注册回调
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
                new PersonThread().start();
            }

            @Override
            public void onServiceDisconnected(ComponentName name) {

            }
        };
        bindService(intent, serviceConnection, BIND_AUTO_CREATE);
    }

    class PersonThread extends Thread {
        @Override
        public void run() {
            for (int i = 0; i < 10; i++) {
                try {
                    iPerson.addPerson("person" + i);
                    Thread.sleep(1000);
                    mHandler.sendEmptyMessage(0);
                } catch (RemoteException e) {
                    e.printStackTrace();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    @Override
    public int getLayoutId() {
        return R.layout.activity_aidl;
    }

    @Override
    public void showLoading() {

    }

    @Override
    public void hideLoading() {

    }

    @Override
    public void refreshView() {

    }

    @Override
    public void onError(String errMessage) {

    }
}

运行结果

Activity客户端结果

Service服务端结果

 

posted @ 2020-08-05 15:41  夜空中最亮的盖子  阅读(2150)  评论(0编辑  收藏  举报