Tauri 开源笔记(3) - 本地数据存储

大家好,我是 codexu,我在撰写一个关于 Tauri 跨端开发的系列文章,分享我在开发笔记 APP NoteGen 过程中所采用技术和遇到的问题。

如果你对本系列有兴趣,可以关注一下往期内容。我将带你从零开始开发一款跨端 APP,帮助你掌握相关知识和技能,即使你不懂 rust,也可以通过官方提供的插件完成大部分的功能开发。

存储分类

在前端开发中,我们很少接触本地数据存储的功能,这部分工作通常是后端来实现,前端使用到的本地存储一般为浏览器提供的方式,例如 localstorage、indexDB 这类,对于文件的操作仅限于对话框打开或保存文件。现在使用使用 Tauri 时,官方提供了相应的插件,可以实现系统级别的存储操作。

我将数据存储分为三类:文件、数据库、简单键值存储,它们所对应的功能和实现方式:

  1. 文件 ,在记录时,需要在本地存放截图、插图的图片文件,在写作时,需要操作 .md 类型的文件。这类文件操作在 Tauri 中统一称为文件系统 ,官方提供了 @tauri-apps/plugin-fs 插件访问文件系统。
  2. 数据库 ,一般前端都是调用后端接口,几乎没有使用过数据库,可能比较陌生,有些同学可能使用过 IndexDB,复杂程度要比 localstorage 复杂的多。但作为一款 APP,数据存在浏览器中可不是一个明智的选择。更好的选择是采用本地数据,我更倾向使用 SQLite,官方提供了 @tauri-apps/plugin-sql 插件,让前端可以通过 sqlx 与 SQL 数据库进行通信,支持 SQLite、MySQL 和 PostgreSQL。
  3. 键值 ,这种很简单,类似平时使用的 localstorage,主要可以应用在系统配置或其他设置的记录上。官方提供了 @tauri-apps/plugin-store 来实现简单、持久的键值存储。

下图为 NoteGen 的数据目录,可以看到有图片、数据库文件、json。

接下来,我将具体介绍如何使用 Tauri 提供的插件进行这三种存储的实际操作。

文件系统

如果你接触过 node,对于这部分应该很好理解。

常见的文件操作包括对文件或文件夹的创建、删除、移动和重命名等。这些操作允许我们读取文件的内容和相关信息,同时也可以判断文件的状态。常见的状态判断包括检查文件或文件夹是否存在。在进行操作之前,一般都会先进行检查,因为对不存在的文件进行操作将会导致错。

通过以下命令进行安装:

shell 复制代码
pnpm tauri add fs

文件系统插件提供的方法,可以查看官方文档。所有的接口参数几乎都类似,一般为 path,options,path 很好理解,就是操作的路径,一般还会带上文件名和后缀。options 中,一般需要填写 baseDir 参数。接下来以 writeTextFile 方法来讲解一下如何使用:

ts 复制代码
function writeTextFile(path, data, options?): Promise<void>

参数:

参数名 类型
path string
data string
options? WriteFileOptions

一般情况下,我们不需要写完整的 path,因为可以通过 options.baseDir 参数来补全前置的路径,baseDir 对应的类型为 BaseDirectory,它是一个 enum 类型,比如常见的 BaseDirectory.AppData 对应了应用程序的数据路径,在 NoteGen 这个项目中,它对应的 path 为:/Users/xu/Library/Application Support/com.codexu.NoteGen,如果你想在这个文件夹下写入一个 demo.md 文件,那么你可以这样调用:

ts 复制代码
import { writeTextFile, BaseDirectory } from '@tauri-apps/plugin-fs'

await writeTextFile('demo.md', 'balabala', {
  baseDir: BaseDirectory.AppData
})

执行后,即可看到 demo.md 已被创建,其他方法也是类似的,理解这一点之后,你便能够轻松上手其他方法了。

SQL

如果没有特殊要求,数据库直接无脑选择 SQLite 即可,因为它是一个轻量级的关系数据库管理系统,它的设计目的是实现自包含、无服务器、零配置的数据库,数据库文件通常只有几百KB,适合作为嵌入式数据库使用。并且 SQLite 是一个跨端的数据库,不论是 PC 还是移动端都完美支持。

首先,在你的 Cargo.toml 文件中添加以下内容来安装插件:

toml 复制代码
[dependencies.tauri-plugin-sql]
features = ["sqlite"] # or "postgres", or "mysql"
version = "2.0.0"

然后使用以下命令安装 SQLite 插件:

shell 复制代码
pnpm tauri add sql

创建和使用数据库

ts 复制代码
import Database from '@tauri-apps/plugin-sql';
const db = await Database.load('sqlite:note.db');

这个命令用于创建或使用数据库。如果数据库不存在,则会自动创建,存储路径为前文提到的 BaseDirectory.AppData

可以看到创建了一个叫 note.db 的文件。

创建表

ts 复制代码
export async function initMarksDb() {
  const db = await Database.load('sqlite:note.db');
  await db.execute(`
    create table if not exists marks (
      id integer primary key autoincrement,
      tagId integer not null,
      type text not null,
      content text default null,
      url text default null,
      desc text default null,
      deleted integer default 0,
      createdAt integer
    )
  `)
}

这里需要自己写 SQL,如果不了解如何写 SQL,可以先去学习一下,或者直接通过 AI 来生成代码。

在 NoteGen 项目中,我在 src/db 下创建了多个文件,来管理数据库,如果有不理解的地方可以参考源码

如果你没有合适的数据库可视化工具,推荐使用 github.com/dbeaver/dbe...,这是一款免费的 SQL 可视化工具,并且他支持 SQLite。

键值

键值存储一般用于系统设置等选项,通常不会存储非常重要的信息,功能类似于浏览器中的 LocalStorage,通过命令安装 Store:

shell 复制代码
pnpm tauri add store

使用方式,由于非常简单,直接引用官方示例:

ts 复制代码
import { Store } from '@tauri-apps/plugin-store';

// Store 会在 JavaScript 绑定时自动加载。
const store = new Store('store.json');

// 设置一个值。
await store.set('some-key', { value: 5 });

// 获取一个值。
const val = await store.get('some-key');
console.log(val); // { value: 5 }

// 您可以在进行更改后手动保存存储
// 否则如上所述,它将在正常退出时保存。
await store.save();

这里有一点修改,是 store.json,这样我们将本地键值对存储为一个 json 文件,另外需要注意的是,这些操作都是异步的。

总结

有了本地文件和数据库的加持,就可以顺利完成后续记录和笔记的数据存储,这对于一个离线应用至为重要。

记得关注点赞Star一波三连,爱你的xu。

相关推荐
GISer_Jing几秒前
DeepSeek 阐述 2025年前端发展趋势
前端·javascript·react.js·前端框架
prince_zxill5 分钟前
RESTful 架构原则及其在 API 设计中的应用
前端·javascript·架构·前端框架·restful
禁默32 分钟前
【学术投稿-2025年计算机视觉研究进展与应用国际学术会议 (ACVRA 2025)】从计算机基础到HTML开发:Web开发的第一步
前端·计算机视觉·html
韦德说1 小时前
【开源事故】77.7K Star 的 Hugo 作者亲自回信!但他第一句话就让我彻底慌了……
后端·开源·go
Chaoran3 小时前
vue3 封装右键菜单组件
前端·javascript
海岸边的破木船3 小时前
为什么Vue3能更好的支持TS
前端
前端on9仔3 小时前
Chrome插件教程:一个小案例给掘金社区添加一键关灯效果
前端·chrome
小狸花子3 小时前
全平台异构文件上传架构设计:打破端侧限制的OSS上传实践
前端
慕斯-ing3 小时前
利用Vue编写一个“计数器”
前端·vue.js·经验分享
暗暗那4 小时前
【腾讯前端面试】纯css画图形
前端·css·面试