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

「简记往来」开发历程系列:数据结构——如何设计收礼和送礼的双向关系

一、普通记账 vs 礼账:数据模型的本质区别

普通记账App的数据模型非常简单:

记录(金额、分类、时间、备注)

统计时只需要按时间或分类汇总即可。

但礼账完全不同。礼账要回答的问题是:

  • “张三累计给了我多少钱?”
  • “我累计给张三多少钱?”
  • “我和张三之间的净额是多少?”

这需要的是双向关系数据模型

二、核心表结构

简记往来的数据库设计围绕两张核心表展开:

联系人表(contacts)

字段类型说明
idstring主键
book_idstring所属账本
namestring标准姓名
tagsarray标签(同事/朋友/亲戚)
created_attimestamp创建时间

记录表(records)

字段类型说明
idstring主键
book_idstring所属账本
contact_idstring关联的联系人ID
typeenumreceive(收礼)/ send(送礼)
amountnumber金额
datestring日期
notestring备注
created_bystring录入人
created_attimestamp创建时间

两张表通过contact_id关联。一个联系人可以有多个记录(收礼和送礼)。

三、双向关系的存储方式

关键问题是:如何用一张表存储“双向关系”?

答案是:用type字段区分方向。

  • type: 'receive'表示“张三给了我钱”
  • type: 'send'表示“我给了张三钱”

查询时,按contact_id聚合,分别统计receivesend的总金额,相减得到净额。

SELECTcontact_id,SUM(CASEWHENtype='receive'THENamountELSE0END)astotal_receive,SUM(CASEWHENtype='send'THENamountELSE0END)astotal_sendFROMrecordsWHEREbook_id=?GROUPBYcontact_id

这就是差额统计的底层逻辑。

四、索引优化方案

6.8万用户、62万笔记录,查询速度怎么保证?

核心索引:

-- 按账本+联系人查询记录CREATEINDEXidx_records_book_contactONrecords(book_id,contact_id)-- 按账本+日期查询CREATEINDEXidx_records_book_dateONrecords(book_id,date)-- 按账本+类型统计CREATEINDEXidx_records_book_typeONrecords(book_id,type)

实际测试中,最复杂的查询(按账本聚合所有联系人的收送礼统计)响应时间稳定在200ms以内。

五、扩展性考虑

这套数据模型还支持几个扩展场景:

  • 多账本:通过book_id隔离,每个账本独立
  • 多人协作:通过created_by记录录入人
  • 数据导出:直接导出contactsrecords的联表查询结果

设计数据模型时想清楚“要回答什么问题”,比一开始就追求“大而全”更重要。简记往来的数据模型从一开始就围绕“人与人的差额”这个核心问题设计,后续所有功能都是在此基础上扩展的,没有做过大的重构。

http://www.jsqmd.com/news/1062802/

相关文章:

  • 校园毕业季风采评比活动|中正投票完整搭建教程 - 投票评选活动
  • Kubernetes ExternalDNS 自动化DNS管理实战
  • Rufus:解决Windows 11安装难题的终极USB启动盘制作工具
  • 开发信息发布平台 APP,开启个性化运营新时代
  • 2026重庆黄金回收实测排行:7证合规商家优选,变现避坑怎么选? - 名奢变现站
  • AestheticNet:融合视觉与语义的图像美学评估新范式
  • OSX-KVM性能飞跃:从虚拟化到原生体验的全面解锁
  • 大语言模型解码策略实战:Beam Search与Tilted Sampling的工程对比与优化
  • 西安整装公司有推荐的吗?3个维度帮你选 - 速递信息
  • ATUC微控制器硬件开发实战:封装、焊接与勘误表避坑指南
  • NSK精机:W2009FS滚珠丝杠技术规范详述
  • 2026 天津全城名表回收渠道,市区环城上门变现指南 - 逸程
  • 2026年天津离婚律师推荐精选:5位攻破财产分割的实力派 - 本地品牌推荐
  • 解放你的塔科夫:SPT-AKI存档编辑器的完全掌控指南
  • C++哈希容器线程安全实战:Metrowerks线程库与并发控制策略
  • Qwen2.5-VL动态分辨率与绝对时间编码技术解析
  • 闲置旧金饰出手防坑技巧,教你筛选广州靠谱二手黄金回收门店 - 开心测评
  • DeepSeek-V4核心技术解析:mHC、CSA、HCA与Muon工程实践
  • 上海冰丰库:上海餐饮配送中心冷藏库动线布局实战指南 - 上海冰丰库制冷
  • 认识Nectin
  • 2026 杭州各区县手表回收攻略 本地人避坑指南各区腕表变现方法详解 - 薛定谔的梨花猫
  • 投票链接怎么做?365评选2026免注册极速版,3分钟一键生成活动 - 微信投票制作
  • Ubuntu 20.04 自建 Python 3.9 编程环境:源码编译与 venv 隔离实战
  • 闲置爱马仕包包回收,2026哈尔滨五大实体门店实力排名优选 - 名奢变现站
  • Why is software operated, maintained, and serviced
  • 基于概率流与Wasserstein度量的动态系统故障检测与恢复控制
  • 语义网络分析:透视3D环境教育游戏玩家认知结构的X光机
  • YOLOv14 vs YOLOv26:60% mAP领跑标准COCO,跨域能力更是降维打击
  • 嵌入式流协议解析:事件驱动通信与触发机制设计
  • 北京本地刑事律师事务所推荐:五家机构办案特色与优势解析 - 品牌2026