使用AIDL实现跨进程接口回掉

栏目: IOS · Android · 发布时间: 5年前

内容简介:同一个进程内实现接口回掉很简单,这里不做叙述,本文主要讲的是跨进程的接口回掉实现方式。有一种跨进程通信的方式就是使用AIDL,但是单纯的AIDL通信只可以实现客户端访问服务端主动获取Binder对象,如果服务端有变化无法及时通知客户端。现在可以通过跨进程的接口回掉来解决服务端发生变化通知客户端的问题。可以使用使用RemoteCallbackList来实现对创建Service,并且在service中定义RemoteCallbackList集合,实现ITestInterface.Stub,在registerCa

同一个进程内实现接口回掉很简单,这里不做叙述,本文主要讲的是跨进程的接口回掉实现方式。有一种跨进程通信的方式就是使用AIDL,但是单纯的AIDL通信只可以实现客户端访问服务端主动获取Binder对象,如果服务端有变化无法及时通知客户端。现在可以通过跨进程的接口回掉来解决服务端发生变化通知客户端的问题。

可以使用使用RemoteCallbackList来实现对 IInterface 的管理。 public class RemoteCallbackList<E extends IInterface>

首先定义两个AIDL文件:

  1. ITestCallBack.aidl
    interface ITestCallBack {
        /**
         * Demonstrates some basic types that you can use as parameters
         * and return values in AIDL.
         */
        void onTagValid(in String tag);
    
    }
    复制代码
  2. ITestInterface.aidl    在注册和反祖册方法中,需要传入ITestCallBack的对象

    interface ITestInterface {
        /**
         * Demonstrates some basic types that you can use as parameters
         * and return values in AIDL.
         */
         boolean isTagValid(in String tag);
    
         void registerCallback(in String tag, in ITestCallBack callback);
    
         void unRegisterCallback(in String tag, in ITestCallBack callback);
    }复制代码

服务端:

创建Service,并且在service中定义RemoteCallbackList集合,实现ITestInterface.Stub,在registerCallback,和unRegisterCallback中,分别将ITestCallBack对象注册和反注册进RemoteCallbackList中。 RemoteCallbackList提供了获取注册进去的IInterface对象方法。

//其实RemoteCallbackList类似于 java 中{@link java.util.Observable},用来批量处理接口回调对象,
//其实如果确保只有一个客户端会bind到这个服务,只需要保存一个IMyAidlInterfaceCallback即可。
//但是如果有多个,强烈推荐使用其实RemoteCallbackList

public void callBack() {
    if (mCallBacks == null) {
        return;
    }
    int num = mCallBacks.beginBroadcast();
    for (int i = 0; i < num; i++) {
        try {
            mCallBacks.getBroadcastItem(i).onTagValid("congratulation callback success " + tag);
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
    //结束后一定要使用finsh,否则下次执行beginBroadcast会抛出IllegalStateException异常
    mCallBacks.finishBroadcast();
}复制代码

在isTagValid中可以调用callBack方法去遍历注册的接口对象, 也可以当服务端有变化时主动调用callBack方法去通知客户端,这样就实现了服务端变化主动通知客户端 。可根据实际方法修改。

在service的onBind方法中,返回ITestInterface.Stub的对象即可,等待客户端绑定服务端。

下面是服务端Service的代码:

public class TestService extends Service {

    private RemoteCallbackList<ITestCallBack> mCallBacks = new RemoteCallbackList<>();
    private String tag = "hy";

    public TestService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        return iTestInterface;
    }

    @Override
    public boolean onUnbind(Intent intent) {
        return super.onUnbind(intent);
    }

    ITestInterface.Stub iTestInterface = new ITestInterface.Stub() {
        @Override
        public boolean isTagValid(String tag) throws RemoteException {
            if (tag.equals(TestService.this.tag)) {
                callBack();
                return true;
            }
            return false;
        }

        @Override
        public void registerCallback(String tag, ITestCallBack callback) throws RemoteException {
            if (null != mCallBacks && null != callback) {
                mCallBacks.register(callback);
            }
        }

        @Override
        public void unRegisterCallback(String tag, ITestCallBack callback) throws RemoteException {
            if (null != mCallBacks && null != callback) {
                mCallBacks.unregister(callback);
            }
        }
    };

    public void callBack() {
        if (mCallBacks == null) {
            return;
        }
        int num = mCallBacks.beginBroadcast();
        for (int i = 0; i < num; i++) {
            try {
                mCallBacks.getBroadcastItem(i).onTagValid("congratulation callback success " + tag);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        mCallBacks.finishBroadcast();
    }
}复制代码

客户端:

客户端首先要做的是绑定服务端,实现AIDL的通信,在客户端创建绑定按钮,解绑按钮,和主动获取信息的通信按钮。在主动获取信息的通信按钮中实现iTestInterface对象的isTagValid方法可以主动去获取服务端的信息(服务端在isTagValid方法中调用了callBack方法)。

客户端代码:

public class MainActivity extends AppCompatActivity {

    private String tag = "hy";
    private ITestInterface iTestInterface;
    private ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iTestInterface = ITestInterface.Stub.asInterface(service);
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            iTestInterface = null;
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        bindService();
        ((Button) findViewById(R.id.buttonregister)).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    iTestInterface.registerCallback(tag, new ITestCallBack.Stub() {
                        @Override
                        public void onTagValid(String tag) throws RemoteException {
                            Log.e("test", "registerCallback: " + tag);
                        }
                    });
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
        });

        ((Button) findViewById(R.id.buttonunregister)).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    iTestInterface.unRegisterCallback(tag, new ITestCallBack.Stub() {
                        @Override
                        public void onTagValid(String tag) throws RemoteException {

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

        ((Button) findViewById(R.id.buttonisvalid)).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    iTestInterface.isTagValid(tag);
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
        });
    }

    private void bindService() {
        Intent intent = new Intent();
        intent.setAction("com.example.heyang.myapplication.TestService");
        intent.setPackage("com.example.heyang.myapplication");
        boolean success = bindService(intent, connection, Context.BIND_AUTO_CREATE);
        if (success) {
            Log.e("test ", "bindService OK");
        } else {
            Log.e("test ", "bindService Fail");
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        unBindeService();
    }

    private void unBindeService() {
        try {
            unbindService(connection);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
复制代码

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

编程珠玑(英文版・第2版)

编程珠玑(英文版・第2版)

[美] Jon Bentley / 人民邮电出版社 / 2010-8 / 39.00元

多年以来,当程序员们推选出最心爱的计算机图书时,《编程珠玑》总是位列前列。正如自然界里珍珠出自细沙对牡蛎的磨砺,计算机科学大师Jon Bentley以其独有的洞察力和创造力,从磨砺程序员的实际问题中凝结出一篇篇不朽的编程“珠玑”。这些文章是《ACM通讯》最受欢迎的专栏文章,最终结集为两部书出版。本书为第一卷,主要讨论计算机科学中最本质的问题:如何正确选择和高效地实现算法。 在书中,作者选取许......一起来看看 《编程珠玑(英文版・第2版)》 这本书的介绍吧!

JSON 在线解析
JSON 在线解析

在线 JSON 格式化工具

随机密码生成器
随机密码生成器

多种字符组合密码