• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# 跨设备文件访问
2<!--Kit: Core File Kit-->
3<!--Subsystem: FileManagement-->
4<!--Owner: @wang_zhangjun; @zhuangzhuang-->
5<!--Designer: @wang_zhangjun; @zhuangzhuang; @renguang1116-->
6<!--Tester: @liuhonggang123; @yue-ye2; @juxiaopang-->
7<!--Adviser: @foryourself-->
8
9分布式文件系统为应用提供了跨设备文件访问的能力,开发者在两个设备上安装同一应用时,通过[基础文件接口](app-file-access.md),可跨设备读写另一个设备上该应用[分布式目录](app-sandbox-directory.md#应用沙箱路径和真实物理路径的对应关系)(/data/storage/el2/distributedfiles/)下的文件。例如:多设备数据流转的场景,设备组网互联之后,设备A上的应用可访问设备B上的同应用分布式目录下的文件,当期望应用文件被其他设备访问时,只需将文件移动到分布式目录即可。
10
11## 开发步骤
12
131. 完成分布式组网。
14   将需要跨设备访问的两个设备登录同一账号,保证设备蓝牙和Wi-Fi功能开启,蓝牙无需互连,Wi-Fi无需接入同一个局域网。
15
162. 授权分布式数据同步权限。
17   分布式数据同步权限的授权方式为user_grant,因此需要调用requestPermissionsFromUser接口,以动态弹窗的方式向用户申请授权。示例中的context的获取方式请参见[获取UIAbility的上下文信息](../application-models/uiability-usage.md#获取uiability的上下文信息)。
18
19   ```ts
20   import { common, abilityAccessCtrl } from '@kit.AbilityKit';
21   import { BusinessError } from '@kit.BasicServicesKit';
22   // 请在组件内获取context,确保this.getUIContext().getHostContext()返回结果为UIAbilityContext
23   let context = this.getUIContext().getHostContext() as common.UIAbilityContext;
24   let atManager = abilityAccessCtrl.createAtManager();
25   try {
26     //以动态弹窗的方式向用户申请授权
27     atManager.requestPermissionsFromUser(context, ['ohos.permission.DISTRIBUTED_DATASYNC']).then((result) => {
28       console.info(`request permission result: ${JSON.stringify(result)}`);
29     }).catch((err: BusinessError) => {
30       console.error(`Failed to request permissions from user. Code: ${err.code}, message: ${err.message}`);
31     })
32   } catch (error) {
33     let err: BusinessError = error as BusinessError;
34     console.error(`Catch err. Failed to request permissions from user. Code: ${err.code}, message: ${err.message}`);
35   }
36   ```
37
383. 访问跨设备文件。
39   同一应用不同设备之间实现跨设备文件访问,只需要将对应的文件放在应用沙箱的分布式目录即可。
40
41   设备A上在分布式目录下创建测试文件,并写入内容。示例中的context的获取方式请参见[获取UIAbility的上下文信息](../application-models/uiability-usage.md#获取uiability的上下文信息)。
42
43   ```ts
44   import { fileIo as fs } from '@kit.CoreFileKit';
45   import { common } from '@kit.AbilityKit';
46   import { BusinessError } from '@kit.BasicServicesKit';
47
48   // 请在组件内获取context,确保this.getUIContext().getHostContext()返回结果为UIAbilityContext
49   let context = this.getUIContext().getHostContext() as common.UIAbilityContext;
50   let pathDir: string = context.distributedFilesDir;
51   // 获取分布式目录的文件路径
52   let filePath: string = pathDir + '/test.txt';
53
54   try {
55     // 在分布式目录下创建文件
56     let file = fs.openSync(filePath, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
57     console.info('Succeeded in creating.');
58     // 向文件中写入内容
59     fs.writeSync(file.fd, 'content');
60     // 关闭文件
61     fs.closeSync(file.fd);
62   } catch (error) {
63     let err: BusinessError = error as BusinessError;
64     console.error(`Failed to openSync / writeSync / closeSync. Code: ${err.code}, message: ${err.message}`);
65   }
66   ```
67
68   设备B主动向设备A发起建链,建链成功后设备B可在分布式目录下读取测试文件。
69   > **说明:**
70   >
71   > 这里通过分布式设备管理的接口获取设备networkId,详见[设备管理接口](../reference/apis-distributedservice-kit/js-apis-distributedDeviceManager.md)。
72
73   ```ts
74   import { fileIo as fs } from '@kit.CoreFileKit';
75   import { common } from '@kit.AbilityKit';
76   import { BusinessError } from '@kit.BasicServicesKit';
77   import { buffer } from '@kit.ArkTS';
78   import { distributedDeviceManager } from '@kit.DistributedServiceKit';
79
80   // 通过分布式设备管理的接口获取设备A的networkId信息
81   let dmInstance = distributedDeviceManager.createDeviceManager("com.example.hap");
82   let deviceInfoList: Array<distributedDeviceManager.DeviceBasicInfo> = dmInstance.getAvailableDeviceListSync();
83   if (deviceInfoList && deviceInfoList.length > 0) {
84     console.info(`Success to get available device list`);
85     let networkId = deviceInfoList[0].networkId;
86     // 定义访问公共文件目录的回调
87     let listeners : fs.DfsListeners = {
88       onStatus: (networkId: string, status: number): void => {
89         console.info('Failed to access public directory');
90       }
91     };
92     // 开始跨设备文件访问
93     fs.connectDfs(networkId, listeners).then(() => {
94       console.info("Success to connect dfs");
95       // 请在组件内获取context,确保this.getUIContext().getHostContext()返回结果为UIAbilityContext
96       let context = this.getUIContext().getHostContext() as common.UIAbilityContext;
97       let pathDir: string = context.distributedFilesDir;
98       // 获取分布式目录的文件路径
99       let filePath: string = pathDir + '/test.txt';
100       try {
101         // 打开分布式目录下的文件
102         let file = fs.openSync(filePath, fs.OpenMode.READ_WRITE);
103         // 定义接收读取数据的缓存
104         let arrayBuffer = new ArrayBuffer(4096);
105         // 读取文件的内容,返回值是读取到的字节个数
106         class Option {
107             public offset: number = 0;
108             public length: number = 0;
109         };
110         let option = new Option();
111         option.length = arrayBuffer.byteLength;
112         let num = fs.readSync(file.fd, arrayBuffer, option);
113         // 打印读取到的文件数据
114         let buf = buffer.from(arrayBuffer, 0, num);
115         console.info('read result: ' + buf.toString());
116         fs.closeSync(file);
117       } catch (error) {
118         let err: BusinessError = error as BusinessError;
119         console.error(`Failed to openSync / readSync. Code: ${err.code}, message: ${err.message}`);
120       }
121     }).catch((error: BusinessError) => {
122       let err: BusinessError = error as BusinessError;
123       console.error(`Failed to connect dfs. Code: ${err.code}, message: ${err.message}`);
124     });
125   }
126   ```
127
1283. B设备访问跨设备文件完成,断开链路。
129
130   ```ts
131   import { BusinessError } from '@kit.BasicServicesKit';
132   import { distributedDeviceManager } from '@kit.DistributedServiceKit';
133   import { fileIo as fs } from '@kit.CoreFileKit';
134
135   // 获取设备A的networkId
136   let dmInstance = distributedDeviceManager.createDeviceManager("com.example.hap");
137   let deviceInfoList: Array<distributedDeviceManager.DeviceBasicInfo> = dmInstance.getAvailableDeviceListSync();
138   if (deviceInfoList && deviceInfoList.length > 0) {
139     console.info(`Success to get available device list`);
140     let networkId = deviceInfoList[0].networkId;
141    // 关闭跨设备文件访问
142     fs.disconnectDfs(networkId).then(() => {
143       console.info("Success to disconnect dfs");
144     }).catch((err: BusinessError) => {
145       console.error(`Failed to disconnect dfs. Code: ${err.code}, message: ${err.message}`);
146     })
147   }
148   ```