当前位置: 首页 > article >正文

鸿蒙系统分布式文件概述、访问、拷贝

1. 分布式文件系统概述

分布式文件系统(hmdfs,HarmonyOS Distributed File System)提供跨设备的文件访问能力,适用于如下场景:

  • 两台设备组网,用户可以利用一台设备上的编辑软件编辑另外一台设备上的文档。

  • 平板保存的音乐,车载系统直接可见并可播放。

  • 户外拍摄的照片,回家打开平板直接访问原设备拍摄的照片。

hmdfs在分布式软总线动态组网的基础上,为网络上各个设备结点提供一个全局一致的访问视图,支持开发者通过基础文件系统的接口进行读写访问,具有高性能、低延时等优点。

2. 设置分布式文件等级

不同设备本身的安全能力差异较大,因此,HarmonyOS提供一套完整的数据分级、设备分级标准,并针对不同设备制定不同的数据流转策略。
官方提供的API:setSecurityLabel

使用方式:

//打开文件
let file = fs.openSync(filePath, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
// 设置文件的数据等级为s0
securityLabel.setSecurityLabel(filePath, 's0').then(() => {
  console.info('Succeeded in setSecurityLabeling.');
  fs.closeSync(file);
}).catch((err: BusinessError) => {
  console.error(`Failed to setSecurityLabel. Code: ${err.code}, message: ${err.message}`);
});

3. 跨设备文件访问

分布式文件系统为应用提供了跨设备文件访问的能力,开发者在两个设备安装同一应用时,通过基础文件接口,可跨设备读写另一个设备该应用分布式文件路径(/data/storage/el2/distributedfiles/)下的文件。例如:多设备数据流转的场景,设备组网互联之后,设备A上的应用可访问设备B同应用分布式路径下的文件,当期望应用文件被其他设备访问时,只需将文件移动到分布式文件路径即可。

使用步骤:

1. 完成组网

将需要跨设备访问的两个设备登录同一账号,保证设备蓝牙和Wi-Fi功能开启,蓝牙无需互连,Wi-Fi无需接入同一个局域网。

2. 访问跨设备文件

同一应用不同设备之间实现跨设备文件访问,只需要将对应的文件放在应用沙箱的分布式文件路径即可。

2.1 设备A在分布式下创建文件:
async createDistributedFile(uri: string) {
    let context = getContext(this) as common.UIAbilityContext; 
    let pathDir: string = context.distributedFilesDir;
    // 获取分布式目录的文件路径
    let filePath: string = pathDir + '/test.txt';

    try {
      // 在分布式目录下创建文件
      let file = fs.openSync(filePath, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
      console.info('Succeeded in createing.');
      // 向文件中写入内容
      fs.writeSync(file.fd, 'content');
      // 关闭文件
      fs.closeSync(file.fd);
    } catch (error) {
      let err: BusinessError = error as BusinessError;
      console.error(`Failed to openSync / writeSync / closeSync. Code: ${err.code}, message: ${err.message}`);
    }
  }
2.2 设备B在分布式路径下读取文件,并保存到本地
/**
   * 拉起picker保存文件
   */
  async saveFile(fileName: string):Promise<string>{
    // fileName = this.tempFilePath;
    PhLog.info('保存的文件名:' + fileName)
    let saveFileUri = ''
    try {
      let DocumentSaveOptions = new picker.DocumentSaveOptions();
      DocumentSaveOptions.newFileNames = [fileName];
      let documentPicker = new picker.DocumentViewPicker();
      await documentPicker.save(DocumentSaveOptions).then( (DocumentSaveResult) => {
        PhLog.info('DocumentViewPicker.save successfully, DocumentSaveResult uri: ' + JSON.stringify(DocumentSaveResult));
        if (DocumentSaveResult !== null && DocumentSaveResult !== undefined) {
          let uri = DocumentSaveResult[0] as string;
          PhLog.info('分布式',`save file uri: ${uri}`);
          this.realFileUri = uri
          this.realSaveFile(fileName)
          saveFileUri = uri
        }
      }).catch((err: BusinessError) => {
        PhLog.error('saveFile-DocumentViewPicker.save failed with err: ' + JSON.stringify(err));
      });
    } catch (err) {
      PhLog.error('saveFile-DocumentViewPicker failed with err: ' + err);
    }

    return saveFileUri
  }

  realFileUri: string = ''

  /**
   * 真正开始保存文件
   */
  async realSaveFile(fileName: string) {
    // let fileName = this.tempFilePath;
    let context = getContext(this) as common.UIAbilityContext; // 获取设备B的UIAbilityContext信息
    let pathDir = context.distributedFilesDir;
    // 获取分布式目录的文件路径
    let filePath = pathDir + '/' + fileName;
    PhLog.info('保存文件时候读取的文件路径:' + filePath)

    //开始写入内容
    let startSaveFileTask: taskpool.Task = new taskpool.Task(startSaveFile3, filePath,this.realFileUri);

    taskpool.execute(startSaveFileTask).then((writtenSize: number) => {
      if (writtenSize > 0) {
        console.info('分布式',`文件写入成功,写入大小: ${writtenSize} 字节`);
        promptAction.showToast({ message: '文件保存成功' })
      } else {
        console.error('分布式','文件写入失败');
      }
      this.deleteDistributedFile(filePath)
    })


@Concurrent
export async function startSaveFile3(filePath: string, realFileUri: string): Promise<number> {
  try {
    // 读取源文件
    const sourceFile = fs.openSync(filePath, fs.OpenMode.READ_ONLY);
    const stats = await fs.stat(filePath);
    const fileSize = stats.size;

    PhLog.info('分布式', `保存的文件: ${filePath} - 大小:${fileSize}`);

    // 目标文件
    const targetFile = fs.openSync(realFileUri, fs.OpenMode.WRITE_ONLY);

    // 分片大小,可根据实际情况调整
    const chunkSize = 1024 * 1024*20; // 1MB
    let offset = 0;
    let totalWritten = 0;

    while (offset < fileSize) {
      const currentChunkSize = Math.min(chunkSize, fileSize - offset);
      const buffer = new ArrayBuffer(currentChunkSize);
      PhLog.info('分布式','01开始读取:'+s)
      // 读取当前分片的数据
      const readLen = fs.readSync(sourceFile.fd, buffer, { offset });
      if (readLen === 0) {
        break;
      }
      // 写入当前分片的数据
      const writeLen = await fs.write(targetFile.fd, buffer, { offset });
      totalWritten += writeLen;
      offset += writeLen;
      // 计算并调用进度回调
      const progress = (totalWritten / fileSize) * 100;
      PhLog.info('分布式', `写入进度: ${progress.toFixed(2)}%`);
    }

    // 关闭文件
    fs.closeSync(sourceFile);
    fs.closeSync(targetFile);

    return totalWritten;
  } catch (error) {
    PhLog.error('分布式', `保存文件时出错: ${error.message}`);
    return -1;
  }
}
到这里就完成了分布式文件的访问、拷贝,整体流程就是:
  • A设备创建分布式文件,写入内容
  • B设备从分布式文件系统目录上读取文件内容
  • 将文件保存到本地

PS:操作文件时候注意添加相关权限


http://www.kler.cn/a/570608.html

相关文章:

  • K8S学习之基础六:k8s中pod亲和性
  • 单例模式(线程案例)
  • 使用Java操作Excel
  • R语言基础| 基本图形绘制
  • 数据集/API 笔记:新加坡风速 API
  • Linux12-UDP\
  • 打造高清3D虚拟世界|零基础学习Unity HDRP高清渲染管线(第一天)
  • go语言逆向-符号恢复
  • iOS逆向工程概述与学习路线图
  • Mysql面试篇笔记:
  • 【Python】在Windows下配置Python最小环境并在React执行Python脚本
  • SpringCloud系列教程(十一):token验证
  • unidbg 实现 JNI 与 Java 交互
  • ffmpeg源码编译支持cuda
  • PyCharm 对接 DeepSeek 大模型的详细操作流程
  • ArcGIS操作:11 计算shp矢量面的质心坐标
  • 【JAVA】ThreadPoolTaskExecutor 线程池学习、后端异步、高并发处理
  • ArcGIS操作:08 计算shp面积并添加到属性表
  • 解释 CSS 盒模型的概念以及如何使用 box-sizing 属性
  • 厦大团队|报告:《读懂大模型概念、技术与应用实践》140 页 PPT(文末附链接下载)