您当前的位置:首页 > 电脑百科 > 数据库 > 百科

如何做好表结构设计?

时间:2023-03-21 11:18:20  来源:微信公众号  作者:王中阳Go

前言

最近有不少前端和测试转Go的朋友在​​交流群​​里聊:如何做表结构设计

大家关心的问题阳哥必须整理出来,希望对大家有帮助。

4个方面

设计数据库表结构需要考虑到以下4个方面:

  1. 数据库范式:通常情况下,我们希望表的数据符合某种范式,这可以保证数据的完整性和一致性。例如,第一范式要求表的每个属性都是原子性的,第二范式要求每个非主键属性完全依赖于主键,第三范式要求每个非主键属性不依赖于其他非主键属性。
  2. 实体关系模型(ER模型):我们需要先根据实际情况画出实体关系模型,然后再将其转化为数据库表结构。实体关系模型通常包括实体、属性、关系等要素,我们需要将它们转化为表的形式。
  3. 数据库性能:我们需要考虑到数据库的性能问题,包括表的大小、索引的使用、查询语句的优化等。
  4. 数据库安全:我们需要考虑到数据库的安全问题,包括表的权限、用户角色的设置等。

设计原则

在设计数据库表结构时,可以参考以下几个优雅的设计原则:

  1. 简单明了:表结构应该简单明了,避免过度复杂化。
  2. 一致性:表结构应该保持一致性,例如命名规范、数据类型等。
  3. 规范化:尽可能将表规范化,避免数据冗余和不一致性。
  4. 性能:表结构应该考虑到性能问题,例如使用适当的索引、避免全表扫描等。
  5. 安全:表结构应该考虑到安全问题,例如合理设置权限、避免SQL注入等。
  6. 扩展性:表结构应该具有一定的扩展性,例如预留字段、可扩展的关系等。

最后,需要提醒的是,优雅的数据库表结构需要在实践中不断迭代和优化,不断满足实际需求和新的挑战。

下面举个示例让大家更好的理解如何设计表结构,如何引入内存,有哪些优化思路:

问题描述

 

图片

 

如上图所示,红框中的视频筛选标签,应该怎么设计数据库表结构?

这是一个很好的应用场景,大家可以先自己想一下。不要着急看我的方案。

需求分析

  1. 可以根据红框的标签筛选视频
  2. 其中综合标签比较特殊,和类型、地区、年份、演员等不一样
  • 综合是根据业务逻辑取值,并不需要入库
  • 类型、地区、年份、演员等需要入库
  1. 设计表结构时要考虑到:
  • 方便获取标签信息,方便把标签信息缓存处理
  • 方便根据标签筛选视频,方便我们写后续的业务逻辑

设计思路

  1. 综合标签可以写到配置文件中(或者写在前端),这些信息不需要灵活配置,所以不需要保存到数据库中
  2. 类型、地区、年份、演员都设计单独的表
  3. 视频表中设计标签表的外键,方便视频列表筛选取值
  4. 标签信息写入缓存,提高接口响应速度
  5. 类型、地区、年份、演员表也要支持对数据排序,方便后期管理维护

表结构设计

视频表

字段

注释

id

视频主键id

type_id

类型表外键id

area_id

地区表外键id

year_id

年份外键id

actor_id

演员外键id

其他和视频直接相关的字段(比如名称)我就省略不写了

类型表

字段

注释

id

类型主键id

name

类型名称

sort

排序字段

地区表

字段

注释

id

类型主键id

name

类型名称

sort

排序字段

年份表

字段

注释

id

类型主键id

name

类型名称

要么是年份正序排列,要么是年份倒序排列,所以不需要sort字段

演员表

字段

注释

id

类型主键id

name

类型名称

sort

排序字段

表结构设计完了,别忘了缓存

缓存策略

首先这些不会频繁更新的筛选条件建议使用缓存:

 

图片

 

  1. 比较常用的就是redis缓存
  2. 再进阶一点,如果你使用Docker,可以把这些配置信息写入docker容器所在物理机的内存中,而不用请求其他节点的redis,进一步降低网络传输带来的耗时损耗
  3. 筛选条件这类配置信息,客户端和服务端可以约定一个更新缓存的机制,客户端直接缓存配置信息,进一步提高性能

列表数据自动缓存

目前很多框架都是支持自动缓存处理的,比如goframe和go-zero,官方文档都做了详细的介绍,不作为本文的重点。

goframe

可以使用ORM链式操作-查询缓存[1]

官方示例:

package main

import (
 "time"

 "github.com/gogf/gf/v2/database/gdb"
 "github.com/gogf/gf/v2/frame/g"
 "github.com/gogf/gf/v2/os/gctx"
)

func main() {
 var (
  db  = g.DB()
  ctx = gctx.New()
 )

 // 开启调试模式,以便于记录所有执行的SQL
 db.SetDebug(true)

 // 写入测试数据
 _, err := g.Model("user").Ctx(ctx).Data(g.Map{
  "name": "xxx",
  "site": "https://xxx.org",
 }).Insert()

 // 执行2次查询并将查询结果缓存1小时,并可执行缓存名称(可选)
 for i := 0; i < 2; i++ {
  r, _ := g.Model("user").Ctx(ctx).Cache(gdb.CacheOption{
   Duration: time.Hour,
   Name:     "vip-user",
   Force:    false,
  }).Where("uid", 1).One()
  g.Log().Debug(ctx, r.Map())
 }

 // 执行更新操作,并清理指定名称的查询缓存
 _, err = g.Model("user").Ctx(ctx).Cache(gdb.CacheOption{
  Duration: -1,
  Name:     "vip-user",
  Force:    false,
 }).Data(gdb.Map{"name": "smith"}).Where("uid", 1).Update()
 if err != nil {
  g.Log().Fatal(ctx, err)
 }

 // 再次执行查询,启用查询缓存特性
 r, _ := g.Model("user").Ctx(ctx).Cache(gdb.CacheOption{
  Duration: time.Hour,
  Name:     "vip-user",
  Force:    false,
 }).Where("uid", 1).One()
 g.Log().Debug(ctx, r.Map())
}

go-zero

DB缓存机制[2]

go-zero缓存设计之持久层缓存[3]

官方文档都做了详细的介绍,不作为本文的重点。

总结

这篇文章介绍了设计数据库表结构应该考虑的4个方面,还有优雅设计的6个原则,举了一个例子分享了我的设计思路,为了提高性能我们也要从多方面考虑缓存问题。

本文抛砖引玉,欢迎大家留言交流。

相关资料

[1]ORM链式操作-查询缓存: https://goframe.org/pages/viewpage.action?pageId=1114346

[2]DB缓存机制: https://go-zero.dev/cn/docs/blog/cache/cache

[3]go-zero缓存设计之持久层缓存: https://go-zero.dev/cn/docs/blog/cache/redis-cache

本文转载自微信公众号「 程序员升级打怪之旅」,作者「王中阳Go」



Tags:表结构   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
前言最近有不少前端和测试转Go的朋友在​​交流群​​里聊:如何做表结构设计?大家关心的问题阳哥必须整理出来,希望对大家有帮助。4个方面设计数据库表结构需要考虑到以下4个方...【详细内容】
2023-03-21  Tags: 表结构  点击:(0)  评论:(0)  加入收藏
一. 前台管理1. 轮播图(5张),首页头部大图,注册登录页左侧大图,logo图,导航条广告二. 后台管理1. 站点用户管理1. 系统用户 SystemUser 列名 数据类型 ...【详细内容】
2021-06-23  Tags: 表结构  点击:(223)  评论:(0)  加入收藏
分享职场生活、职场攻略、程序员创业资源,为一线开发者提供优质内容--创建表create table tb_user( `id` int(11) not null auto_increment, user_name varchar(100),...【详细内容】
2020-09-22  Tags: 表结构  点击:(171)  评论:(0)  加入收藏
1.前言作为一名技术人员,少不了接触数据库,那么当遇到一个旧项目数据库表比较多,怎么快速的导出表结构,形成文档方便后续跟进呢?今天我给大家安利一个好用的工具mysql_markdown。...【详细内容】
2020-06-11  Tags: 表结构  点击:(875)  评论:(0)  加入收藏
一、pt-online-schema-change介绍pt-online-schema-change是percona公司开发的一个工具,在percona-toolkit包里面可以找到这个功能,它可以在线修改表结构。而避免被锁表的情况...【详细内容】
2019-10-21  Tags: 表结构  点击:(495)  评论:(0)  加入收藏
1. redis中的链表在redis中链表的应用非常广泛,例如列表键的底层实现之一就是链表。而且,在redis中的链表结构被实现成为双向链表,因此,在头部和尾部进行的操作就会非常快。通过...【详细内容】
2019-08-05  Tags: 表结构  点击:(477)  评论:(0)  加入收藏
▌简易百科推荐
前言最近有不少前端和测试转Go的朋友在​​交流群​​里聊:如何做表结构设计?大家关心的问题阳哥必须整理出来,希望对大家有帮助。4个方面设计数据库表结构需要考虑到以下4个方...【详细内容】
2023-03-21  王中阳Go  微信公众号  Tags:表结构   点击:(0)  评论:(0)  加入收藏
在开发可视化项目的过程中往往涉及到可视化图表, 我们看到的很多酷炫的报表, 大屏, 都用了非常多的图表, 接下来我和大家分享一些比较流行的开源免费的图表库。前言 在开...【详细内容】
2023-03-20  趣谈前端  今日头条  Tags:可视图表库   点击:(4)  评论:(0)  加入收藏
随着互联网应用的不断发展,应用程序越来越复杂,同时面对的数据量也越来越大,对于性能的要求也越来越高。缓存技术的应用已经成为了许多互联网应用的必备技术之一。SpringBoot作...【详细内容】
2023-03-16  Java编程世界  今日头条  Tags:缓存   点击:(3)  评论:(0)  加入收藏
我们使用了Druid、MyBatis Plus和Sharding JDBC来实现分表功能。其中,Druid是一个高性能的数据库连接池,MyBatis Plus是一个基于MyBatis的ORM框架,Sharding JDBC是一个分库分表...【详细内容】
2023-03-13  摆脱格子衫  今日头条  Tags:分表   点击:(18)  评论:(0)  加入收藏
如果是 app 端的查询,不建议设计多字段排序,因为在多字段排序的环境下,服务端在进行多条件的过滤查询时,可能会把有效的数据给过滤掉,如果无法避开,尽量将多个排序字段合并到一个...【详细内容】
2023-03-13  鸭血粉丝Tang  微信公众号  Tags:数据重复   点击:(6)  评论:(0)  加入收藏
当我们的数据达到一定的量级之后,单表甚至单库都无法支撑之时,那么,便会涉及到分库分表。分库分表的方式有多种,开源的解决方案也很多,都是围绕客户端和代理两种模式来处理的。...【详细内容】
2023-03-10  自学编程之道  今日头条  Tags:分库分表   点击:(4)  评论:(0)  加入收藏
上篇文章介绍了Mongo读取数据的策略(​​MongoDB读数据策略​​),主要是readconcern、readpreference两参数,其中readconcern作用于服务端,决定了什么时候能读取到数据;readprefer...【详细内容】
2023-03-09  云数据库技术  微信公众号  Tags:MongoDB   点击:(9)  评论:(0)  加入收藏
数据库在每个企业都占据的非常重要的位置,它存储着一个公司的命脉。对于数据库的管理也非常的复杂多变,很多公司都有专业的DBA团队在管理。但是,不论有无DBA,对数据库的操作都...【详细内容】
2023-03-07  运维开发故事  微信公众号  Tags:SQL   点击:(10)  评论:(0)  加入收藏
01引言数据库管理员(DBA)承担着保障生产数据库稳定运行的职责,在完成生产变更、事件处置等工作的同时,还应该在哪些方面持续提升自身能力呢?本文从银行DBA的视角,谈一谈“DBA的...【详细内容】
2023-03-07  匠心独运维妙维效  微信公众号  Tags:DBA   点击:(10)  评论:(0)  加入收藏
一、背景在《​​# 分布式锁上-初探​​》中有提到一个分布式锁应具备的功能特点中有避免死锁这一条:如果某个客户端获得锁之后处理时间超过最大约定时间,或者持锁期间内发生...【详细内容】
2023-03-07  架构染色  微信公众号  Tags:分布式锁   点击:(8)  评论:(0)  加入收藏
站内最新
站内热门
站内头条