HarmonyOS-鸿蒙app开发 —基于java_流转_HarmonyOS端发起多端协同_多端协同开发指导

HarmonyOS-鸿蒙app开发 —基于java_流转_HarmonyOS端发起多端协同_多端协同开发指导

场景介绍

开发者在应用FA中通过调用流转任务管理服务、分布式任务调度的接口,实现多端协同。

主要流程如下:

设备A上的应用FA向流转任务管理服务注册一个流转回调。

Alt1-系统推荐流转:系统感知周边有可用设备后,主动为用户提供可选择流转的设备信息,并在用户完成设备选择后回调onConnected通知应用FA开始流转,将用户选择的设备B的设备信息提供给应用FA。

Alt2-用户手动流转:系统在用户手动点击流转图标后,通过showDeviceList通知流转任务管理服务,被动为用户提供可选择交互的设备信息,并在用户完成设备选择后回调onConnected通知应用FA开始流转,将用户选择的设备B的设备信息提供给应用FA。

设备A上的应用FA需要初始化分布式任务调度能力。

设备A上的应用FA通过调用分布式任务调度的能力,向设备B的应用发起多端协同。应用FA需要自己管理流转状态,将流转状态从IDLE迁移到CONNECTING,并上报到流转任务管理服务。

开发者根据Ability模板及意图的不同,通过组合以下能力生成多端协同的业务:启动远程FA、启动远程PA、连接远程PA。这些能力都需要指定待连接设备的信息。

下面以设备A(本地设备)和设备B(远端设备)为例,进行场景介绍:

startAbility设备A启动设备B的FA:在设备A上通过本地应用提供的启动按钮,启动设备B上对应的FA。例如:设备A控制设备B打开相册,只需开发者在启动FA时指定打开相册的意图即可。

startAbility设备A启动设备B的PA:在设备A上通过本地应用提供的启动按钮,启动设备B上指定的PA。例如:开发者在启动远程服务时通过意图指定音乐播放服务,即可实现设备A启动设备B音乐播放的能力。

connectAbility设备A连接设备B的PA:在设备A上通过本地应用提供的连接按钮,连接设备B上指定的PA。连接后,通过其他功能相关按钮实现控制对端PA的能力。通过连接关系,开发者可以实现跨设备的同步服务调度,实现如大型计算任务互助等价值场景。

设备A上的应用FA启动或连接设备B的应用FA/PA后,应用将流转状态从CONNECTING迁移到CONNECTED,并上报到流转任务管理服务。

用户通过设备A的流转任务管理界面结束流转。用户点击结束任务后,流转任务管理服务回调onDisconnected通知应用FA取消流转。

设备A上的应用通过调用分布式任务调度的能力,终止和设备B的多端协同。

disconnectAbility设备A与设备B的PA断开连接:将之前已连接的PA断开连接。

stopAbility设备A关闭设备B的PA:关闭设备B上指定的PA。

应用关闭分布式任务调度能力。

应用将流转状态从CONNECTED迁移到IDLE,并上报到流转任务管理服务。

应用向流转任务管理服务注销流转回调。

接口说明

流转任务管理服务提供的注册、解注册、显示设备列表、上报业务状态是实现多端协同的前提。开发者使用分布式服务平台提供的连接和断开连接PA、启动远程FA、启动和关闭PA的能力,可实现自定义的多端协同体验。

表1 IContinuationRegisterManager接口功能介绍

接口名

描述

void register(String bundleName, ExtraParams parameter, IContinuationDeviceCallback deviceCallback, RequestCallback requestCallback)

注册并连接到流转任务管理服务,并获取对应的注册token。

参数说明:

bundleName,必填,String类型,本应用包名。

params,可选,ExtraParams类型,系统推荐流转时所需的过滤条件。如对智能推荐无特殊要求,可与showDeviceList接口的过滤条件保持一致。若需关闭系统推荐流转,ExtraParams中jsonParams传入{“isTurnOffRecommend”:true}即可。

deviceCallback,可选,设备选择列表中选择设备后的回调,返回选择设备ID。

requestCallback,可选,注册请求回调,返回应用注册后的token。

ExtraParams说明:

devType,可选,待连接的设备类型。可取值(”00E”手机、”011″平板、”06D”手表、”09C”智慧屏),如”devType”:[“011”]。若为空,则支持则待连接的设备类型为手机、平板、手表和智慧屏。

targetBundleName,可选,目标应用包名。若为空,则目标应用包名与方法入参bundleName本应用包名相同。

description,可选,ability描述,用于在设备列表页面展示

jsonParams,可选,拓展参数。用于设备过滤等。示例:

{“filter”:{“commonFilter”: {“system”:{“harmonyVersion”:”2.0.0″},”groupType”: “1”,”curComType”: 0x00000004, “faFilter”:”{\”targetBundleName\”:\”com.xxx.yyy\”}”}},”transferScene”:0,”isTurnOffRecommend”:false,”remoteAuthenticationDescription”: “拉起HiVision扫描弹框描述”,”remoteAuthenticationPicture”:””}

jsonParams说明:

system:可选,String类型,待连接设备的HarmonyOS版本号,目标设备对应的harmonyVersion>=该值,如”system”:{“harmonyVersion”:”2.0.0″}。

groupType:可选,String类型。为空时,不要求是否同帐号。本设备和待连接设备的帐号是否要求一致,可取值(1要求同帐号、1|256不要求同帐号),如”groupType”:”1″。

curComType:可选,int类型。为空时,不要求在同一局域网下。本设备和待连接设备是否需要在同一局域网下,可取值(0x00000004需要在同一局域网下,0x00030004不需要在同一局域网下),如”curComType”:0x00000004。

faFilter:可选,String类型,目标应用包名。为空时,不做版本兼容性检查。要做版本兼容性检查时,需要传入目标应用包名。使用faFilter,设备需登录帐号。多端设备可使用不同帐号,但不支持无账号使用faFilter。

transferScene:可选,int类型,默认值为0。可取值为:0-单选协同场景,设备选择面板为单选面板,设备间互斥,流转成功设备面板自动消失,流转失败设备面板不消失,且系统会维护设备的流转状态,如流转成功设备面板消失后再打开设备面板,会显示之前设备流转成功的状态;1-单选迁移场景,设备选择面板为单选面板,设备间互斥,流转成功设备面板自动消失,流转失败设备面板不消失,且系统不会维护设备的流转状态,如流转成功设备面板消失后再打开设备面板,设备流转状态为未流转;2-多选协同场景,设备选择面板为多选面板,流转成功或失败设备面板不消失,设备间不互斥,系统会维护设备的流转状态。

isTurnOffRecommend:可选,boolean类型,默认为false,true:关闭系统推荐流转,false:开启系统推荐流转。

remoteAuthenticationDescription:可选,String类型,跨账号/无账号设备进行扫码认证时,拉起对端HiVision扫描弹框描述。register接口无需传该参数,showDeviceList可按需传参。

remoteAuthenticationPicture:可选,String类型,跨账号/无账号设备进行扫码认证时,拉起对端HiVision扫描弹框显示的图片。需将图片对应的byte[]转String, Base64.encodeToString(mBuff,Base64.DEFAULT)。register接口无需传该参数,showDeviceList可按需传参。

注册后,通过RequestCallback的onResult回调知道执行是否成功,返回值为<0时为失败;其他情况,表示成功,并返回本次流转任务的唯一标识token。

当用户选择设备后,通过deviceCallback定义的onConnected回调获取设备的deviceID、类型、名称。

void unregister(int token, RequestCallback requestCallback)

从流转任务管理服务解注册,传入注册时获取的token进行解注册。

执行后,通过RequestCallback的onResult回调知道执行是否成功。

void updateConnectStatus(int token, String deviceId, int status, RequestCallback requestCallback)

通知流转任务管理服务更新当前用户程序的连接状态,并在流转任务管理服务界面展示给用户。token、deviceId参数来自于注册流转任务管理服务的回调。status参数可以为IDLE、CONNECTING、CONNECTED、DIS_CONNECTING。如果有错误,需要上报errorCode。

执行后,通过RequestCallback的onResult回调知道执行是否成功。

void showDeviceList(int token, ExtraParams parameter, RequestCallback requestCallback)

显示组网内可选择设备列表信息。该接口提供手动显示设备列表的能力,parameter参数可以指定设备过滤的条件,用于手动多端协同,支持的过滤条件与register接口相同。token参数来自于注册流转任务管理服务的回调。

执行后,通过RequestCallback的onResult回调知道执行是否成功。

void disconnect()

在应用退出时,主动调用断开和流转任务管理服务的连接。

表2 IContinuationDeviceCallback接口功能介绍

接口名

描述

void onConnected(ContinuationDeviceInfo deviceInfo)

当用户从选择设备列表选择设备时调用,返回设备ID、设备类型和设备名称供开发者使用。

void onDisconnected(String deviceId)

当流转任务管理服务断开连接设备时调用。

表3 RequestCallback接口功能介绍

接口名

描述

void onResult(int result)

与流转任务管理服务交互成功时调用。当作为注册流转任务管理服务的回调对象时,注册成功后给用户程序返回对应的token。

表4 Ability/AbilitySlice接口功能介绍

接口名

描述

IContinuationRegisterManager getContinuationRegisterManager()

获取流转任务管理服务注册服务管理类,可以与流转任务管理服务进行交互,包括注册/解注册,更新设备连接状态,显示可选择设备列表等。

void startAbility(Intent intent)

提供启动指定设备上FA和PA的能力,Intent中指定待启动FA/PA的设备deviceId、bundleName和abilityName。若不指定设备deviceId,则无法跨设备调用PA。类似地,在启动FA时,也需要开发者指定启动FA的设备deviceId、bundleName和abilityName。

分布式服务平台还会提供与上述功能相对应的断开远程PA的连接和关闭远程PA的接口,相关的参数与连接、启动的接口类似。

boolean connectAbility(Intent intent, IAbilityConnection conn)

提供连接指定设备上PA的能力,Intent中指定待连接PA的设备deviceId、bundleName和abilityName。当连接成功后,通过在conn定义的onAbilityConnectDone回调中获取对端PA的服务代理,两者的连接关系则由conn维护。

boolean stopAbility(Intent intent)

提供关闭远程PA的能力,Intent中指定待关闭PA的设备deviceId、bundleName和abilityName。

void terminateAbility()

提供关闭当前FA/PA的能力,调用该接口将开始销毁调用的FA/PA。

void disconnectAbility​(IAbilityConnection conn)

提供断开远程PA连接能力,conn需要指定为connectAbility连接指定设备上PA时,传入的IAbilityConnection对象。

表5 IAbilityConnection接口功能介绍

接口名

描述

void onAbilityConnectDone(ElementName element, IRemoteObject remote, int resultCode)

当连接Service服务成功时调用,可使用返回的IRemoteObject对象与连接的Service服务通信。

void onAbilityDisconnectDone(ElementName element, int resultCode)

当已连接的Service服务被异常关闭时调用。

表6 DeviceManager接口功能介绍

接口名

描述

void initDistributedEnvironment(String deviceId, IInitCallBack callback)

初始化分布式服务平台。其中deviceId指明设备Id;callback指明初始化分布式环境状态回调,参考IInitCallBack接口描述。

void unInitDistributedEnvironment(String deviceId, IInitCallBack callback)

不再使用分布式服务平台。

表7 IInitCallBack接口功能介绍

接口名

描述

void onInitSuccess(String deviceId)

成功回调。

void onInitFailure(String deviceId, int errorCode)

失败回调。

约束与限制

每个应用注册流转任务管理服务的Ability数量上限为5个,后续新增注册的Ability会将最开始注册的覆盖。

startAbility、connectAbility中跨设备传递的intent数据大小限制200KB以内。

不支持使用connectAbility触发远端PA的免安装。

connectAbility中跨设备传递的remoteObject数据大小限制200KB以内。

多端协同要求HarmonyOS 2.0以上版本才能支持,注册到流转任务管理服务时jsonParams中需要增加{“harmonyVersion”:”2.0.0″}过滤条件。

stopAbility不支持两个设备之间分别登录不同的帐号,也就是要求多个设备是同帐号。

开发步骤

完成环境搭建,在DevEco Studio中,选择手机设备,Empty Feature Ability(Java)模板创建项目,在项目自动创建的MainAbility中实现IAbilityContinuation接口。

public class MainAbility extends Ability implements IAbilityContinuation {

private static final int DOMAIN_ID = 0xD001100;

private static final HiLogLabel LABEL_LOG = new HiLogLabel(3, DOMAIN_ID, “MainAbility”);

 

@Override

public void onStart(Intent intent) {

super.onStart(intent);

super.setMainRoute(MainAbilitySlice.class.getName());

}

 

// 为了方便演示,不在Ability实现流转逻辑,具体逻辑在AbilitySlice中实现

@Override

public boolean onStartContinuation() {

HiLog.info(LABEL_LOG, “onStartContinuation called”);

return true;

}

 

@Override

public boolean onSaveData(IntentParams saveData) {

HiLog.info(LABEL_LOG, “onSaveData called”);

return true;

}

 

@Override

public boolean onRestoreData(IntentParams restoreData) {

HiLog.info(LABEL_LOG, “onRestoreData called”);

return true;

}

 

@Override

public void onCompleteContinuation(int result) {

HiLog.info(LABEL_LOG, “onCompleteContinuation called”);

}

}

在AbilitySlice中实现一个用于控制基础功能的页面,以下演示代码逻辑都将在AbilitySlice中实现,代码示例如下:

public class MainAbilitySlice extends AbilitySlice {

private static final int DOMAIN_ID = 0xD000F00;

private static final HiLogLabel LABEL_LOG = new HiLogLabel(3, DOMAIN_ID, “MainAbilitySlice”);

 

@Override

public void onStart(Intent intent) {

super.onStart(intent);

// 开发者可以自行进行界面设计

// 为按钮设置统一的背景色

// 例如通过PositionLayout可以实现简单界面

PositionLayout layout = new PositionLayout(this);

LayoutConfig config = new LayoutConfig(LayoutConfig.MATCH_PARENT, LayoutConfig.MATCH_PARENT);

layout.setLayoutConfig(config);

ShapeElement buttonBg = new ShapeElement();

buttonBg.setRgbColor(new RgbColor(0, 125, 255));

super.setUIContent(layout);

}

 

@Override

public void onInactive() {

super.onInactive();

}

 

@Override

public void onActive() {

super.onActive();

}

 

@Override

public void onBackground() {

super.onBackground();

}

 

@Override

public void onForeground(Intent intent) {

super.onForeground(intent);

}

 

@Override

public void onStop() {

super.onStop();

}

}

在FA对应的config.json中声明多设备协同访问的权限:ohos.permission.DISTRIBUTED_DATASYNC。在config.json中的配置如下:

{

“module”: {

“reqPermissions”: [

{

“name”: “ohos.permission.DISTRIBUTED_DATASYNC”,

“reason”: “need”,

“usedScene”: {

“ability”: [

“MainAbility”

],

“when”: “inuse”

}

}

],

}

}

此外,还需要在FA的onStart()中,调用requestPermissionsFromUser()方法向用户申请权限,代码示例如下:

public class MainAbility extends Ability implements IAbilityContinuation {

@Override

public void onStart(Intent intent) {

super.onStart(intent);

// 开发者显示声明需要使用的权限

requestPermissionsFromUser(new String[]{“ohos.permission.DISTRIBUTED_DATASYNC”}, 0);

}

}

设置流转任务管理服务回调函数,注册流转任务管理服务,管理流转的目标设备,同时需要在流转结束时解注册流转任务管理服务。

public class MainAbilitySlice extends AbilitySlice {

// 当前应用包名

private String BUNDLE_NAME = “XXX.XXX.XXX”;

// 流转应用包名

private String REMOTE_BUNDLE_NAME = “XXX.XXX.XXX”;

// 流转FA名称

private String REMOTE_FA_NAME = “XXX.XXX.XXX.XXXAbility”;

// 流转PA名称

private String REMOTE_PA_NAME = “XXX.XXX.XXX.XXXAbility”;

// 注册流转任务管理服务后返回的Ability token

private int abilityToken;

// 用户在设备列表中选择设备后返回的设备ID

private String selectDeviceId;

// 用户是否已发起可拉回流转流程

private boolean isReversibly = false;

// 获取流转任务管理服务管理类

private IContinuationRegisterManager continuationRegisterManager;

// 设置初始化分布式环境的回调

private IInitCallback iInitCallback = new IInitCallback() {

@Override

public void onInitSuccess(String deviceId) {

HiLog.info(LABEL_LOG, “device id success: “ + deviceId);

}

 

@Override

public void onInitFailure(String deviceId, int errorCode) {

HiLog.info(LABEL_LOG, “device id failed: “ + deviceId + “errorCode: “ + errorCode);

}

};

// 设置流转任务管理服务设备状态变更的回调

private IContinuationDeviceCallback callback = new IContinuationDeviceCallback() {

@Override

public void onConnected(ContinuationDeviceInfo deviceInfo) {

// 在用户选择设备后设置设备ID

selectDeviceId = deviceInfo.getDeviceId();

try {

// 初始化分布式环境

DeviceManager.initDistributedEnvironment(selectDeviceId, iInitCallback);

} catch (RemoteException e) {

HiLog.info(LABEL_LOG, “initDistributedEnvironment failed”);

}

//更新选择设备后的流转状态

continuationRegisterManager.updateConnectStatus(abilityToken, selectDeviceId, DeviceConnectState.CONNECTED.getState(), null);

}

 

@Override

public void onDisconnected(String deviceId) {

}

};

// 设置注册流转任务管理服务回调

private RequestCallback requestCallback = new RequestCallback() {

@Override

public void onResult(int result) {

abilityToken = result;

}

};

 

@Override

public void onStart(Intent intent) {

continuationRegisterManager = getContinuationRegisterManager();

}

 

@Override

public void onStop() {

super.onStop();

// 解注册流转任务管理服务

continuationRegisterManager.unregister(abilityToken, null);

// 断开流转任务管理服务连接

continuationRegisterManager.disconnect();

}

为不同功能设置相应的控制按钮。

// 建议开发者按照自己的界面进行按钮设计,示例代码仅供参考

private static final int OFFSET_X = 100;

private static final int OFFSET_Y = 100;

private static final int ADD_OFFSET_Y = 150;

private static final int BUTTON_WIDTH = 800;

private static final int BUTTON_HEIGHT = 100;

private static final int TEXT_SIZE = 50;

private int offsetY = 0;

 

private Button btnShowDeviceList;

private Button btnStartRemote;

private Button btnStopRemote;

private Button btnConnectRemotePA;

private Button btnControlRemotePA;

private Button btnDisconnectRemotePA;

 

private Button createButton(String text, ShapeElement buttonBg) {

Button button = new Button(this);

button.setContentPosition(OFFSET_X, OFFSET_Y + offsetY);

offsetY += ADD_OFFSET_Y;

button.setWidth(BUTTON_WIDTH);

button.setHeight(BUTTON_HEIGHT);

button.setTextSize(TEXT_SIZE);

button.setTextColor(Color.YELLOW);

button.setText(text);

button.setBackground(buttonBg);

return button;

}

 

// 按照顺序在PositionLayout中依次添加按钮的示例

private void addComponents(PositionLayout linear, ShapeElement buttonBg) {

// 构建显示注册流转任务管理服务的按钮

Button btnRegister = createButton(“register”, buttonBg);

btnRegister.setClickedListener(mRegisterListener);

linear.addComponent(btnRegister);

 

// 构建显示设备列表的按钮

btnShowDeviceList = createButton(“ShowDeviceList”, buttonBg);

btnShowDeviceList.setClickedListener(mShowDeviceListListener);

linear.addComponent(btnShowDeviceList);

 

// 构建远程启动FA/PA的按钮

btnStartRemote = createButton(“StartRemote”, buttonBg);

btnStartRemote.setClickedListener(mStartRemoteListener);

linear.addComponent(btnStartRemote);

 

// 构建远程关闭PA的按钮

btnStopRemote = createButton(“StopRemote”, buttonBg);

btnStopRemote.setClickedListener(mStopRemoteListener);

linear.addComponent(btnStopRemote);

 

// 构建连接远程PA的按钮

btnConnectRemotePA = createButton(“ConnectRemotePA”, buttonBg);

btnConnectRemotePA.setClickedListener(mConnectRemotePAListener);

linear.addComponent(btnConnectRemotePA);

 

// 构建控制连接PA的按钮

btnControlRemotePA = createButton(“ControlRemotePA”, buttonBg);

btnControlRemotePA.setClickedListener(mControlPAListener);

linear.addComponent(btnControlRemotePA);

 

// 构建与远程PA断开连接的按钮

btnDisconnectRemotePA = createButton(“DisconnectRemotePA”, buttonBg);

btnDisconnectRemotePA.setClickedListener(mDisconnectRemotePAListener);

linear.addComponent(btnDisconnectRemotePA);

}

 

@Override

public void onStart(Intent intent) {

//添加功能按钮布局

addComponents(layout, buttonBg);

super.setUIContent(layout);

}

注册流转任务管理服务。

// 注册流转任务管理服务

private Component.ClickedListener mRegisterListener = new Component.ClickedListener() {

@Override

public void onClick(Component arg0) {

HiLog.info(LABEL_LOG, “register call.”);

//增加过滤条件

ExtraParams params = new ExtraParams();

String[] devTypes = new String[]{ExtraParams.DEVICETYPE_SMART_PAD, ExtraParams.DEVICETYPE_SMART_PHONE};

params.setDevType(devTypes);

String jsonParams = “{‘filter’:{‘commonFilter’:{‘system’:{‘harmonyVersion’:’2.0.0′},’groupType’:’1|256′,’curComType’: 0x00030004,’faFilter’:'{\”localVersionCode\”:1,\”localMinCompatibleVersionCode\”:2,\”targetBundleName\”: \”com.xxx.yyy\”}’}},’transferScene’:0,’remoteAuthenticationDescription’: ‘拉起HiVision扫描弹框描述’,’remoteAuthenticationPicture’:”}”;

params.setJsonParams(jsonParams);

continuationRegisterManager.register(BUNDLE_NAME, params, callback, requestCallback);

}

};

通过流转任务管理服务提供的showDeviceList()接口获取选择设备列表,用户选择设备后在IContinuationDeviceCallback回调中获取设备ID。

// 显示设备列表,获取设备信息

private ClickedListener mShowDeviceListListener = new ClickedListener() {

@Override

public void onClick(Component arg0) {

// 设置过滤设备类型

ExtraParams params = new ExtraParams();

String[] devTypes = new String[]{ExtraParams.DEVICETYPE_SMART_PAD, ExtraParams.DEVICETYPE_SMART_PHONE};

params.setDevType(devTypes);

String jsonParams = “{‘filter’:{‘commonFilter’:{‘system’:{‘harmonyVersion’:’2.0.0′},’groupType’:’1|256′,’curComType’: 0x00030004,’faFilter’:'{\”localVersionCode\”:1,\”localMinCompatibleVersionCode\”:2,\”targetBundleName\”: \”com.xxx.yyy\”}’}},’transferScene’:0,’remoteAuthenticationDescription’: ‘拉起HiVision扫描弹框描述’,’remoteAuthenticationPicture’:”}”;

params.setJsonParams(jsonParams);

// 显示选择设备列表

continuationRegisterManager.showDeviceList(abilityToken, params, null);

}

};

为启动远程FA/PA的按钮设置点击回调,实现启动FA/PA和关闭远程PA的能力。

说明

步骤7步骤8分别介绍不同的流转方式,开发者根据实际需要选择相应的开发方式。

// 启动远程FA/PA

private ClickedListener mStartRemoteListener = new ClickedListener() {

@Override

public void onClick(Component arg0) {

if (selectDeviceId != null) {

// 通过showDeviceList获取指定目标设备deviceId

// 指定待启动FA/PA的bundleName和abilityName

// 设置分布式标记,表明当前涉及分布式能力

Operation operation = new Intent.OperationBuilder()

.withDeviceId(selectDeviceId)

.withBundleName(REMOTE_BUNDLE_NAME)

.withAbilityName(REMOTE_FA_NAME)

.withFlags(Intent.FLAG_ABILITYSLICE_MULTI_DEVICE)

.build();

Intent startIntent = new Intent();

startIntent.setOperation(operation);

// 通过AbilitySlice包含的startAbility接口实现跨设备启动FA/PA

startAbility(startIntent);

} else {

btnStartRemote.setText(“StartRemote selectDeviceId is null”);

}

}

};

 

// 关闭远程PA

private ClickedListener mStopRemoteListener = new ClickedListener() {

@Override

public void onClick(Component arg0) {

if (selectDeviceId != null) {

// 通过showDeviceList获取指定目标设备deviceId

// 指定待关闭PA的bundleName和abilityName

// 设置分布式标记,表明当前涉及分布式能力

Operation operation = new Intent.OperationBuilder()

.withDeviceId(selectDeviceId)

.withBundleName(REMOTE_BUNDLE_NAME)

.withAbilityName(REMOTE_PA_NAME)

.withFlags(Intent.FLAG_ABILITYSLICE_MULTI_DEVICE)

.build();

Intent stopIntent = new Intent();

stopIntent.setOperation(operation);

// 通过AbilitySlice包含的stopAbility接口实现跨设备关闭PA

stopAbility(stopIntent);

} else {

btnStopRemote.setText(“StopRemote selectDeviceId is null”);

}

}

};

需要注意,目标FA/PA需要在config.json中设置visible为true。visible标签表示Ability是否可以被其他应用调用,默认为false,即只允许同应用(同appid)访问;如需被其他应用访问,需要将其设置为true,同时建议在目标FA/PA中添加自定义权限,控制访问范围,防止被其他应用随意访问。

在config.json中的配置如下:

{

“module”: {

“abilities”: [

{

“visible”: true

}

]

}

}

设备A连接设备B侧的PA,利用连接关系调用该PA执行特定任务,以及断开连接。

// 当连接完成时,用来提供管理已连接PA的能力

private MyRemoteProxy mProxy = null;

// 用于管理连接关系

private IAbilityConnection mConn = new IAbilityConnection() {

@Override

public void onAbilityConnectDone(ElementName element, IRemoteObject remote, int resultCode) {

// 跨设备PA连接完成后,会返回一个序列化的IRemoteObject对象

// 通过该对象得到控制远端服务的代理

mProxy = new MyRemoteProxy(remote);

btnConnectRemotePA.setText(“connectRemoteAbility done”);

}

 

@Override

public void onAbilityDisconnectDone(ElementName element, int resultCode) {

// 当已连接的远端PA异常关闭时,会触发该回调

// 支持开发者按照返回的错误信息进行PA生命周期管理

disconnectAbility(mConn);

}

};

仅通过启动/关闭两种方式对PA进行调度无法应对需长期交互的场景,因此,系统向开发者提供了跨设备PA连接及断开连接的能力。为了对已连接PA进行管理,开发者需要实现一个满足IAbilityConnection接口的连接状态检测实例,通过该实例可以对连接及断开连接完成时设置具体的处理逻辑,例如:获取控制对端PA的代理等。进一步为了使用该代理跨设备调度PA,开发者需要在本地及对端分别实现对外接口一致的代理。一个具备加法能力的代理示例如下:

// 以连接提供加法计算能力的PA为例。为了提供跨设备连接能力,需要在本地发起连接侧和对端被连接侧分别实现代理

// 发起连接侧的代理示例如下:

public class MyRemoteProxy implements IRemoteBroker {

private static final int ERR_OK = 0;

private static final int COMMAND_PLUS = IRemoteObject.MIN_TRANSACTION_ID;

private static final String DESCRIPTOR = “com.XXX.DESCRIPTOR”;

private final IRemoteObject remote;

 

public MyRemoteProxy(IRemoteObject remote) {

this.remote = remote;

}

 

@Override

public IRemoteObject asObject() {

return remote;

}

 

public int plus(int a, int b) throws RemoteException {

MessageParcel data = MessageParcel.obtain();

MessageParcel reply = MessageParcel.obtain();

// option不同的取值,决定采用同步或异步方式跨设备控制PA

// 本例需要同步获取对端PA执行加法的结果,因此采用同步的方式,即MessageOption.TF_SYNC

// 具体MessageOption的设置,可参考相关API文档

MessageOption option = new MessageOption(MessageOption.TF_SYNC);

data.writeInterfaceToken(DESCRIPTOR);

data.writeInt(a);

data.writeInt(b);

 

try {

remote.sendRequest(COMMAND_PLUS, data, reply, option);

int errCode = reply.readInt();

if (errCode != ERR_OK) {

throw new RemoteException();

}

int result = reply.readInt();

return result;

} finally {

data.reclaim();

reply.reclaim();

}

}

}

此外,对端待连接的PA需要实现对应的客户端,代码示例如下所示:

// 以计算加法为例,对端实现的客户端如下

public class MyRemote extends RemoteObject implements IRemoteBroker{

private static final int ERR_OK = 0;

private static final int ERROR = –1;

private static final int COMMAND_PLUS = IRemoteObject.MIN_TRANSACTION_ID;

private static final String DESCRIPTOR = “com.XXX.DESCRIPTOR”;

 

public MyRemote() {

super(“MyService_Remote”);

}

 

@Override

public IRemoteObject asObject() {

return this;

}

 

@Override

public boolean onRemoteRequest(int code, MessageParcel data, MessageParcel reply, MessageOption option) {

String token = data.readInterfaceToken();

if (!DESCRIPTOR.equals(token)) {

reply.writeInt(ERROR);

return false;

}

 

if (code != COMMAND_PLUS) {

reply.writeInt(ERROR);

return false;

}

 

int value1 = data.readInt();

int value2 = data.readInt();

int sum = value1 + value2;

reply.writeInt(ERR_OK);

reply.writeInt(sum);

return true;

}

}

对端除了要实现如上所述的客户端外,待连接的PA还需要作如下修改:

// 为了返回给连接方可调用的代理,需要在该PA中实例化客户端,例如作为该PA的成员变量

private MyRemote remote = new MyRemote();

// 当该PA接收到连接请求时,即将该客户端转化为代理返回给连接发起侧

@Override

protected IRemoteObject onConnect(Intent intent) {

super.onConnect(intent);

return remote.asObject();

}

创建远程连接目标PA的步骤可参考创建Service。需要注意,目标PA需要在config.json中设置visible为true。visible标签表示Ability是否可以被其他应用调用,默认为false,即只允许同应用(同appid)访问;如需被其他应用访问,需要将其设置为true,同时建议在目标PA中添加自定义权限,控制访问范围,防止被其他应用随意访问。

在config.json中的配置如下:

{

“module”: {

“abilities”: [

{

“visible”: true

}

]

}

}

完成上述步骤后,可以通过点击事件实现连接、利用连接关系控制PA以及断开连接等行为,代码示例如下:

// 连接远程PA

private ClickedListener mConnectRemotePAListener = new ClickedListener() {

@Override

public void onClick(Component arg0) {

if (selectDeviceId != null) {

// 指定待连接PA的bundleName和abilityName

// 设置分布式标记,表明当前涉及分布式能力

Operation operation = new Intent.OperationBuilder()

.withDeviceId(selectDeviceId)

.withBundleName(REMOTE_BUNDLE_NAME)

.withAbilityName(REMOTE_PA_NAME)

.withFlags(Intent.FLAG_ABILITYSLICE_MULTI_DEVICE)

.build();

Intent connectPAIntent = new Intent();

connectPAIntent.setOperation(operation);

// 通过AbilitySlice包含的connectAbility接口实现跨设备连接PA

connectAbility(connectPAIntent, mConn);

}

}

};

 

// 控制已连接PA执行加法

private ClickedListener mControlPAListener = new ClickedListener() {

@Override

public void onClick(Component arg0) {

if (mProxy != null) {

int ret = –1;

try {

ret = mProxy.plus(10, 20);

} catch (RemoteException e) {

HiLog.error(LABEL_LOG, “ControlRemotePA error”);

}

btnControlRemotePA.setText(“ControlRemotePA result = “ + ret);

}

}

};

 

// 与远程PA断开连接

private ClickedListener mDisconnectRemotePAListener = new ClickedListener() {

@Override

public void onClick(Component arg0) {

// 按钮复位

btnConnectRemotePA.setText(“ConnectRemotePA”);

btnControlRemotePA.setText(“ControlRemotePA”);

disconnectAbility(mConn);

}

};

说明

通过连接/断开连接远程PA,与跨设备PA建立长期的管理关系。例如在本例中,通过连接关系得到远程PA的控制代理后,实现跨设备计算加法并将结果返回到本地显示。在实际开发中,开发者可以根据需要实现多种分布式场景,例如:跨设备位置/电量等信息的采集、跨设备计算资源互助等。

🚀 如未找到文章请搜索栏搜素 | Ctrl+D收藏本站 | 联系邮箱:15810050733@qq.com