1.大致流程

  • 在服务端声明远程服务接口IRemoteService.aidl,并声明回调接口ICallback.aidl
  • 在服务端实现远程服务接口IRemoteService.Stub
  • 使用RemoteCallbackList保存回调接口列表
  • 发布服务
  • 在客户端实现回调接口 ICallback.Stub
  • 绑定服务,注册回调接口
  • 调用服务
  • 远程服务从RemoteCallbackList中找到回调,然后调用.

2.服务端

2.1 声明服务接口IRemoteService.aidl

 // IRemoteService.aidl
 package com.example.ee.aidl;

 import com.example.ee.aidl.ICallback;

 interface IRemoteService {

     int getPid();
     oneway void fun1();
     void registerCallback(ICallback cb);
     void unregisterCallback(ICallback cb);
 }

2.2 声明回调接口ICallback.aidl

 // ICallback.aidl
 package com.example.ee.aidl;

 interface ICallback {
     oneway void fun2();
 }

2.3 实现服务,用 RemoteCallbackList 保存回调接口.

 package com.example.ee.aidl;

 import android.app.Service;
 import android.content.Intent;
 import android.os.IBinder;
 import android.os.Process;
 import android.os.RemoteCallbackList;
 import android.os.RemoteException;
 import android.support.annotation.Nullable;
 import android.util.Log;

 public class RemoteService extends Service {
     final String TAG = "RemoteService";

     private RemoteBinder binder = new RemoteBinder();
     private RemoteCallbackList<ICallback>  callbackList = new RemoteCallbackList<>();

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

     class RemoteBinder extends IRemoteService.Stub{

         void callback(){
             int count = callbackList.beginBroadcast();
             try {
                 ; i < count; i++) {
                     callbackList.getBroadcastItem(i).fun2();
                 }
             } catch (RemoteException e) {
                 Log.e(TAG, e.getMessage());
             } finally {
             }
             callbackList.finishBroadcast();
         }

         @Override
         public int getPid() throws RemoteException {
             return Process.myPid();
         }

         @Override
         public void fun1() throws RemoteException {
             Log.d(TAG, "fun1 in remote service ");
             callback();
         }

         @Override
         public void registerCallback(ICallback cb) throws RemoteException {
             callbackList.register(cb);
         }

         @Override
         public void unregisterCallback(ICallback cb) throws RemoteException {
             callbackList.unregister(cb);
         }
     }
 }

2.4 发布服务

 <?xml version="1.0" encoding="utf-8"?>
 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
     package="com.example.ee.aidl">

     <application
         android:allowBackup="false"
         android:icon="@mipmap/ic_launcher"
         android:label="@string/app_name"
         android:roundIcon="@mipmap/ic_launcher_round"
         android:supportsRtl="true"
         android:name=".AidlApp"
         android:theme="@style/AppTheme">
         <!--<meta-data
             android:name="APP_CHANNEL"
             android:value="${APP_CHANNEL}" />
 -->
         <activity android:name=".MainActivity">
             <intent-filter>
                 <action android:name="android.intent.action.MAIN" />

                 <category android:name="android.intent.category.LAUNCHER" />
             </intent-filter>
         </activity>
         <service android:name=".RemoteService" android:process=".RemoteService" android:exported="true"/>
     </application>

 </manifest>

3.客户端

3.1 实现回调接口

     private ICallback callback  = new ICallback.Stub(){

         @Override
         public void fun2() throws RemoteException {
             Log.d(TAG, "fun2 in client ");
         }
     };

3.2 绑定远程服务,注册回调

     private void bindRemoteService(){
         connection = new ServiceConnection() {
             @Override
             public void onServiceConnected(ComponentName name, IBinder service) {
                 remoteService = IRemoteService.Stub.asInterface(service);
                 try {
                     remoteService.registerCallback(callback);
                 } catch (RemoteException e) {
                     e.printStackTrace();
                 }
             }

             @Override
             public void onServiceDisconnected(ComponentName name) {

             }
         };

         Intent intent = new Intent(this,RemoteService.class);
         boolean ret = bindService(intent,connection, BIND_AUTO_CREATE);
         if (!ret ){
             Log.e(TAG, "bindRemoteService: bind failed.");
         }
     }

3.3 调用远程服务

     public void onClick(View view){
         if(view.getId() == R.id.btn_call_remote){
             AidlApp app = (AidlApp) getApplication();
             if (app.remoteService != null){
                 try {
                     int pid = app.remoteService.getPid();
                     Log.d(TAG, "onServiceConnected: remote pid = " + pid);
                     app.remoteService.fun1();
                 } catch (RemoteException e) {
                     e.printStackTrace();
                 }
             }else {
                 Snackbar.make(view,"remote service unbinded",Snackbar.LENGTH_SHORT).show();
             }
         }
     }

4. 远程服务调用回调

     void callback(){
             int count = callbackList.beginBroadcast();
             try {
                 ; i < count; i++) {
                     callbackList.getBroadcastItem(i).fun2();
                 }
             } catch (RemoteException e) {
                 Log.e(TAG, e.getMessage());
             } finally {
             }
             callbackList.finishBroadcast();
     }

5.下载

  https://gitee.com/xi/RemoteCallbackList.git

AIDL(2):服务端回调客户端的更多相关文章

  1. 解决有关flask-socketio中服务端和客户端回调函数callback参数的问题(全网最全)

    由于工作当中需要用的flask_socketio,所以自己学习了一下如何使用,查阅了有关文档,当看到回调函数callback的时候,发现文档里都描述的不太清楚,最后终于琢磨出来了,分享给有需要的朋友 ...

  2. WCF心跳判断服务端及客户端是否掉线并实现重连接

    WCF心跳判断服务端及客户端是否掉线并实现重连接 本篇文章将通过一个实例实现对WCF中针对服务端以及客户端是否掉线进行判断:若掉线时服务器或客户端又在线时将实现自动重连:将通过WCF的双工知识以及相应 ...

  3. Java的oauth2.0 服务端与客户端的实现

    oauth原理简述 oauth本身不是技术,而是一项资源授权协议,重点是协议!Apache基金会提供了针对Java的oauth封装.我们做Java web项目想要实现oauth协议进行资源授权访问,直 ...

  4. oauth2.0服务端与客户端搭建

    oauth2.0服务端与客户端搭建 - 推酷 今天搭建了oauth2.0服务端与客户端.把搭建的过程记录一下.具体实现的功能是:client.ruanwenwu.cn的用户能够通过 server.ru ...

  5. 使用WebSocket实现服务端和客户端的通信

    开发中经常会有这样的使用场景.如某个用户在一个数据上做了xx操作, 与该数据相关的用户在线上的话,需要实时接收到一条信息. 这种可以使用WebSocket来实现. 另外,对于消息,可以定义一个类进行固 ...

  6. Java 断点下载(下载续传)服务端及客户端(Android)代码

    原文: Java 断点下载(下载续传)服务端及客户端(Android)代码 - Stars-One的杂货小窝 最近在研究断点下载(下载续传)的功能,此功能需要服务端和客户端进行对接编写,本篇也是记录一 ...

  7. asp.net获取服务端和客户端信息

    asp.net获取服务端和客户端信息 获取服务器名:Page.Server.ManchineName获取用户信息:Page.User 获取客户端电脑名:Page.Request.UserHostNam ...

  8. python thrift 服务端与客户端使用

    一.简介 thrift是一个软件框架,用来进行可扩展且跨语言的服务的开发.它结合了功能强大的软件堆栈和代码生成引擎,以构建在 C++, Java, Python, PHP, Ruby, Erlang, ...

  9. IE8下服务端获取客户端文件的路径为C:/fakePath问题的解决方案

    上一篇文章上提到,IE8下服务端获取客户端文件的路径时,会变成C:/fakePath问题,于是乎通过文件路径去获得文件大小就失败了. 上网搜了一下,主要原因是IE8因为安全考虑,在上传文件时屏蔽了真实 ...

随机推荐

  1. 打造一套UI与后台并重.net通用权限管理系统

    一.前言 从进行到软件开发这个行业现在已经有几年了,在整理出这个套开发框架之前自己做了不少重复造轮子的事.每次有新的项目总是要耗费不少时间在UI.权限和系统通用模块上面,自己累得要死,老板还骂没效率. ...

  2. Laravel框架中的数据库CURD操作、连贯操作、链式操作的用法

    Laravel是一套简洁.优雅的PHP Web开发框架(PHP Web Framework).它可以让你从面条一样杂乱的代码中解脱出来:它可以帮你构建一个完美的网络APP,而且每行代码都可以简洁.富于 ...

  3. Floyd-Warshall求图中任意两点的最短路径

    原创 除了DFS和BFS求图中最短路径的方法,算法Floyd-Warshall也可以求图中任意两点的最短路径. 从图中任取两点A.B,A到B的最短路径无非只有两种情况: 1:A直接到B这条路径即是最短 ...

  4. 微软和Google的盈利模式对比分析

    一: 微软和Google是世界上最成功科技巨头之一,但他们之间却有着不同的产品和业务,二者的盈利方式也各有不同,本文将分析和探讨的二者盈利模式的异同. 微软的盈利模式 在1975年由大学肄业的Bill ...

  5. 接上一篇,Springcloud使用feignclient远程调用服务404 ,为什么去掉context-path后,就能够调通

    一.问题回顾 如果application.properties文件中配置了 #项目路径 server.servlet.context-path=/pear-cache-service 则feigncl ...

  6. Kotlin 函数

    至于什么函数,在计算机里面就是一个密闭的执行程序的代码块(个人理解) 我们先来看看什么是函数 fun main(agrs : Array<String>) { println(" ...

  7. loj #6235. 区间素数个数

    #6235. 区间素数个数 题目描述 求 1∼n 1\sim n1∼n 之间素数个数. 输入格式 一行一个数 n nn . 输出格式 一行一个数,表示答案. 样例 样例输入 10 样例输出 4 样例解 ...

  8. python基础之内置函数(一)

    内建函数都在 _builtins_ 里面 (1)abs() 取绝对值 adb(-10) 10 (2)bool()将参数转换成布尔型,返回值是True或False 参数是数字时,0返回False,其他任 ...

  9. eclipse的一些快捷键记录

    查看所有快捷键:Ctrl + Shift + L Ctrl + Shift + F:格式化(关闭搜狗输入法的所有快捷键) Alt + /:智能提示,自动补全 Ctrl + / :添加单行注释(取消:C ...

  10. echart在jsp中使用另外的方法

    var chartOutChar = null; var option1 = { tooltip: { trigger: 'axis' }, toolbox: { feature: { dataVie ...