Go书签工具采用树形结构建模嵌套文件夹,用指针操作避免拷贝;构建倒排索引加速关键词检索;SQLite手动建表持久化,事务保障一致性;CLI聚焦add/find/export高频操作,修改即更新索引、读取走内存缓存树。

用 Go 写一个轻量、高效、可本地运行的书签管理工具,核心不在功能堆砌,而在数据结构选得准、检索路径理得清。重点不是“存多少”,而是“找得快”和“改得稳”。
用嵌套结构表达书签层级关系
浏览器书签天然带文件夹嵌套(如 编程/Go/官方文档),硬扁平化会丢失语义。推荐用树形结构建模:
- 每个节点含 Name、URL(为空表示是文件夹)、Children 切片
- 根节点设为虚拟文件夹(如 "Bookmarks Bar"),避免多根歧义
- 用指针传递节点(*Node)避免深拷贝,增删改操作直接作用于原结构
构建倒排索引加速关键词检索
全量遍历树找“golang interface”太慢。可在内存中维护一个 map[string][]*Node:
- 键是分词后的关键词(用 strings.FieldsFunc(title, unicode.IsSpace) 粗粒度切分)
- 值是匹配该词的所有节点指针(支持标题、URL、文件夹路径三字段联合索引)
- 插入/更新书签时同步更新索引;删除时用 slice 删除技巧(如
nodes = append(nodes[:i], nodes[i+1:]...))清理引用
用 SQLite 做持久化,但绕过 ORM
不依赖 GORM 或 sqlx,直接用 database/sql + sqlite3 驱动,手动设计两张表:
立即学习“go语言免费学习笔记(深入)”;
- folders 表:id, name, parent_id(NULL 表示根)
- bookmarks 表:id, title, url, folder_id, sort_order(整型序号,支持拖拽排序)
- 读取时用 JOIN 查询 + 递归构建树;写入时用事务包裹文件夹+书签批量操作,保证一致性
命令行交互保持简洁可控
不用 Web UI,用 spf13/cobra 搭建 CLI,聚焦高频动作:
- bkm add -t "Go maps" -u "https://go.dev/blog/maps" -p "编程/Go"(自动创建中间文件夹)
- bkm find "interface method"(查索引后去重输出,显示完整路径如 编程/Go/语言规范 → interface method)
- bkm export json > bk.json(导出为标准书签 HTML 或 JSON,兼容 Chrome 导入)
基本上就这些。结构清晰了,检索自然快;持久化踏实了,数据就不怕丢;CLI 守住边界,反而更容易迭代。不复杂但容易忽略的是:每次修改都触发索引更新、每次读取都走缓存树而非反复查库——这才是 Go 做小工具的舒服感。










