AIDL(2):服务端回调客户端
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):服务端回调客户端的更多相关文章
- 解决有关flask-socketio中服务端和客户端回调函数callback参数的问题(全网最全)
由于工作当中需要用的flask_socketio,所以自己学习了一下如何使用,查阅了有关文档,当看到回调函数callback的时候,发现文档里都描述的不太清楚,最后终于琢磨出来了,分享给有需要的朋友 ...
- WCF心跳判断服务端及客户端是否掉线并实现重连接
WCF心跳判断服务端及客户端是否掉线并实现重连接 本篇文章将通过一个实例实现对WCF中针对服务端以及客户端是否掉线进行判断:若掉线时服务器或客户端又在线时将实现自动重连:将通过WCF的双工知识以及相应 ...
- Java的oauth2.0 服务端与客户端的实现
oauth原理简述 oauth本身不是技术,而是一项资源授权协议,重点是协议!Apache基金会提供了针对Java的oauth封装.我们做Java web项目想要实现oauth协议进行资源授权访问,直 ...
- oauth2.0服务端与客户端搭建
oauth2.0服务端与客户端搭建 - 推酷 今天搭建了oauth2.0服务端与客户端.把搭建的过程记录一下.具体实现的功能是:client.ruanwenwu.cn的用户能够通过 server.ru ...
- 使用WebSocket实现服务端和客户端的通信
开发中经常会有这样的使用场景.如某个用户在一个数据上做了xx操作, 与该数据相关的用户在线上的话,需要实时接收到一条信息. 这种可以使用WebSocket来实现. 另外,对于消息,可以定义一个类进行固 ...
- Java 断点下载(下载续传)服务端及客户端(Android)代码
原文: Java 断点下载(下载续传)服务端及客户端(Android)代码 - Stars-One的杂货小窝 最近在研究断点下载(下载续传)的功能,此功能需要服务端和客户端进行对接编写,本篇也是记录一 ...
- asp.net获取服务端和客户端信息
asp.net获取服务端和客户端信息 获取服务器名:Page.Server.ManchineName获取用户信息:Page.User 获取客户端电脑名:Page.Request.UserHostNam ...
- python thrift 服务端与客户端使用
一.简介 thrift是一个软件框架,用来进行可扩展且跨语言的服务的开发.它结合了功能强大的软件堆栈和代码生成引擎,以构建在 C++, Java, Python, PHP, Ruby, Erlang, ...
- IE8下服务端获取客户端文件的路径为C:/fakePath问题的解决方案
上一篇文章上提到,IE8下服务端获取客户端文件的路径时,会变成C:/fakePath问题,于是乎通过文件路径去获得文件大小就失败了. 上网搜了一下,主要原因是IE8因为安全考虑,在上传文件时屏蔽了真实 ...
随机推荐
- JSP内置对象与servlet对应关系
隐式对象 说明 out 转译后对应JspWriter对象,其内部关联一个PringWriter对象 request 转译后对应HttpServletRequest/ServletRequest对象 r ...
- RegExp正则表达式对象
JavaScript的RegExp对象有两种创建方式,一种是字面量,一种是对象. var r = /pattern/attributes或者new RegExp(pattern, attributes ...
- 装饰(Decorator)模式
一. 装饰(Decorator)模式 装饰(Decorator)模式又名包装(Wrapper)模式[GOF95].装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案. 二. 装饰模式 ...
- ulua
unity窗口 lua -> Gen LuaWrap + Binder 生成Wrap文件 此类脚本是对Unity中常用的组件进行二次包装 Lua运行后 会把Wrap文件加载到Lua运行环境中 使 ...
- owinAuthorize
Nuget包获取 Install-Package Microsoft.AspNet.WebApi.Owin -Version 5.1.2 Install-Package Microsoft.Owin. ...
- .NET和C#的版本历史
维基百科页面:https://en.wikipedia.org/wiki/.NET_Framework_version_history Versionnumber CLRversion Release ...
- Unity性能优化专题---腾讯牛人分享经验
这里从三个纬度来分享下内存的优化经验:代码层面.贴图层面.框架设计层面. 一.代码层面. 1.foreach. Mono下的foreach使用需谨慎.频繁调用容易触及堆上限,导致GC过早触发,出现卡顿 ...
- leecode刷题(6)-- 两个数组的交集II
leecode刷题(6)-- 两个数组的交集II 两个数组的交集II 描述: 给定两个数组,编写一个函数来计算它们的交集. 示例: 输入: nums1 = [1,2,2,1], nums2 = [2, ...
- ceph_osd故障检测
1. 当前monitor可以通过3种途径检测到osd离线 1) Osd自主上报 2) Osd通过投票的方式(满足一下条件之一,mon会将osd标记为down) a) ...
- requests库和urllib包对比
python中有多种库可以用来处理http请求,比如python的原生库:urllib包.requests类库.urllib和urllib2是相互独立的模块,python3.0以上把urllib和ur ...