鸿蒙数据库表中的数据如何导出为Excel存到系统下载目录

【背景知识】

fs.write:将数据写入文件,使用promise异步回调。

DocumentViewPicker:文件选择器对象,用来支撑选择和保存各种格式文档。

逗号分隔值(Comma-Separated Values,CSV,有时也称为字符分隔值,因为分隔字符也可以不是逗号),其文件以纯文本形式存储表格数据(数字和文本)。

【解决方案】

构造Excel表格对应的CSV文件数据。

使用fs.write将数据写入Excel文件。

使用DocumentViewPicker将文件保存到用户文件目录下。

示例代码如下:

typescript 复制代码
import fs from '@ohos.file.fs';
import { picker } from '@kit.CoreFileKit';
import { BusinessError } from '@kit.BasicServicesKit';
import { hilog } from '@kit.PerformanceAnalysisKit';

const DOMAIN = 0x0000;

interface Product {
  name: string;
  price: number;
  quantity: number;
}
// 导出数据库时,此处换为查表得到的list即可
let products: Product[] = [
  { name: 'Prod,uct1', price: 10.0, quantity: 5 },
  { name: 'Pr"oduc"t2', price: 15.0, quantity: 3 }
];

@Entry
@Component
struct ExcelDemo {
  convertToCSV(data: Product[]): string {
    let headers = ["名称", "价格", "数量"];
    let fields = ["name", "price", "quantity"];
    let csvContent = headers.join("\t") + "\n";
    data.forEach((product) => {
      let rowData = fields.map(field => {
        let value: string = Object(product)[field].toString() || ''; // 获取字段值
        // 移除换行符
        value = value.toString().replace(/[\n\r]/g, ' ');
        // 如果值中包含逗号、引号,则用双引号包裹,并且转义双引号
        if (typeof value === 'string' && (value.includes(',') || value.includes('"'))) {
          let temp = value.replace(/"/g, '""');
          value = `"${temp}"`;
        }
        return value;
      });
      csvContent += rowData.join("\t") + "\n";
    });
    return csvContent;
  }

  build() {
    RelativeContainer() {
      Button('write excel').onClick(() => {
        // 定义文件路径,这里假设要保存到应用的内部存储某个目录下,可根据实际需求调整。
        try {
          let uris: Array<string> = [];
          // 创建文件管理器选项实例
          const documentSaveOptions = new picker.DocumentSaveOptions();
          documentSaveOptions.newFileNames = ["Test20.xlsx"];
          documentSaveOptions.fileSuffixChoices = ['文档|.xlsx', '.xlsx'];
          const documentViewPicker = new picker.DocumentViewPicker(getContext(this));
          // 用户选择目标文件夹,用户选择与文件类型相对应的文件夹,即可完成文件保存操作。保存成功后,返回保存文档的URI。
          documentViewPicker.save(documentSaveOptions).then((documentSaveResult: Array<string>) => {
            uris = documentSaveResult;
            let file = fs.openSync(uris[0], fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
            // 构建CSV数据
            let csvData = this.convertToCSV(products);
            fs.write(file.fd, csvData).then((writeLen: number) => {
              hilog.info(DOMAIN, 'testTag', 'write data to file succeed and size is:', writeLen);
            }).catch((err: BusinessError) => {
              hilog.error(DOMAIN, 'testTag',
                "write data to file failed with error message: ", err.message, ", error code: ", err.code);
            }).finally(() => {
              fs.closeSync(file);
            });

            hilog.info(DOMAIN, 'testTag', 'documentViewPicker.save to file succeed and uris are:', uris);
          }).catch((err: BusinessError) => {
            hilog.error(DOMAIN, 'testTag',
              `Invoke documentViewPicker.save failed, code is ${err.code}, message is ${err.message}`);
          })
        } catch (e) {
          hilog.error(DOMAIN, 'testTag', "documentViewPicker.save error: ", JSON.stringify(e));
        }
      })
    }
    .height('100%')
    .width('100%')
  }
}
相关推荐
oh,huoyuyan7 分钟前
【界面案例】火语言RPA读取Excel文件,循环写入界面表格
excel·rpa
wang60212521818 分钟前
为什么不采用级联删除而选择软删除
数据库·oracle
变形侠医1 小时前
比 Kettle 快2倍的 Java ETL 开源库:Etl-engine
数据库
soft20015251 小时前
从一次增删改操作开始:彻底理解 MySQL Buffer Pool 的地位与作用
数据库·mysql
feathered-feathered1 小时前
Redis基础知识+RDB+AOF(面试)
java·数据库·redis·分布式·后端·中间件·面试
whm27772 小时前
Visual Basic 建立数据库
开发语言·数据库·visual studio
檀越剑指大厂2 小时前
【数据库系列】倒排索引(Inverted Index)
数据库·inverted-index
闲人编程2 小时前
权限系统设计与RBAC实现
java·网络·数据库·rbac·权限·codecapsule
讯方洋哥2 小时前
HarmonyOS应用开发——应用状态
华为·harmonyos
ujainu2 小时前
鸿蒙与Flutter:全场景开发的技术协同与价值
flutter·华为·harmonyos