>* 表内的每一行都应该被唯一的标识(有唯一键)。
* 表内不应该存储依赖于其他键的非键信息。
遵守 3nf 标准的数据库具有以下特点:有一组表专门存放通过键连接起来的关联数据。比方说,某个存放客户及其有关定单的 3nf 数据库就可能有两个表:customer 和 order。order 表不包含定单关联客户的任何信息,但表内会存放一个键值,该键指向 customer 表里包含该客户信息的那一行。
更高层次的标准化也有,但更标准是否就一定更好呢?答案是不一定。事实上,对某些项目来说,甚至就连 3nf 都可能给数据库引入太高的复杂性。

为了效率的缘故,对表不进行标准化有时也是必要的,这样的例子很多。曾经有个开发餐饮分析软件的活就是用非标准化表把查询时间从平均 40 秒降低到了两秒左右。虽然我不得不这么做,但我绝不把数据表的非标准化当作当然的设计理念。而具体的操作不过是一种派生。所以如果表出了问题重新产生非标准化的表是完全可能的。

microsoft visual foxpro 报表技巧
如果你正在使用 microsoft visual foxpro,你可以用对用户友好的字段名来代替编号的名称:比如用 customer name 代替 txtcnam。这样,当你用向导程序 [wizards,台湾人称为‘精灵’] 创建表单和报表时,其名字会让那些不是程序员的人更容易阅读。不活跃或者不采用的指示符
增加一个字段表示所在记录是否在业务中不再活跃挺有用的。不管是客户、员工还是其他什么人,这样做都能有助于再运行查询的时候过滤活跃或者不活跃状态。同时还消除了新用户在采用数据时所面临的一些问题,比如,某些记录可能不再为他们所用,再删除的时候可以起到一定的防范作用。使用角色实体定义属于某类别的列[字段]
在需要对属于特定类别或者具有特定角色的事物做定义时,可以用角色实体来创建特定的时间关联关系,从而可以实现自我文档化。
这里的含义不是让 person 实体带有 title 字段,而是说,为什么不用 person 实体和 person_type 实体来描述人员呢?比方说,当 john smith, engineer 提升为 john smith, director 乃至最后爬到 john smith, cio 的高位,而所有你要做的不过是改变两个表 person 和 person_type 之间关系的键值,同时增加一个日期/时间字段来知道变化是何时发生的。这样,你的 person_type 表就包含了所有 person 的可能类型,比如 associate、engineer、director、cio 或者 ceo 等。
还有个替代办法就是改变 person 记录来反映新头衔的变化,不过这样一来在时间上无法跟踪个人所处位置的具体时间。采用常用实体命名机构数据
组织数据的最简单办法就是采用常用名字,比如:person、organization、address 和 phone 等等。当你把这些常用的一般名字组合起来或者创建特定的相应副实体时,你就得到了自己用的特殊版本。开始的时候采用一般术语的主要原因在于所有的具体用户都能对抽象事物具体化。
有了这些抽象表示,你就可以在第 2 级标识中采用自己的特殊名称,比如,person 可能是 employee、spouse、patient、client、customer、vendor 或者 teacher 等。同样的,organization 也可能是 mycompany、mydepartment、competitor、hospital、warehouse、government 等。最后 address 可以具体为 site、location、home、work、client、vendor、corporate 和 fieldoffice 等。
采用一般抽象术语来标识“事物”的类别可以让你在关联数据以满足业务要求方面获得巨大的灵活性,同时这样做还可以显著降低数据存储所需的冗余量。用户来自世界各地
在设计用到网络或者具有其他国际特性的数据库时,一定要记住大多数国家都有不同的字段格式,比如邮政编码等,有些国家,比如新西兰就没有邮政编码一说。数据重复需要采用分立的数据表
如果你发现自己在重复输入数据,请创建新表和新的关系。每个表中都应该添加的 3 个有用的字段
* drecordcreationdate,在 vb 下默认是 now(),而在 sql server 下默认为 getdate()
* srecordcreator,在 sql server 下默认为 not null default user
* nrecordversion,记录的版本标记;有助于准确说明记录中出现 null 数据或者丢失数据的原因对地址和电话采用多个字段
描述街道地址就短短一行记录是不够的。address_line1、address_line2 和 address_line3 可以提供更大的灵活性。还有,电话号码和邮件地址最好拥有自己的数据表,其间具有自身的类型和标记类别。

过分标准化可要小心,这样做可能会导致性能上出现问题。虽然地址和电话表分离通常可以达到最佳状态,但是如果需要经常访问这类信息,或许在其父表中存放“首选”信息(比如 customer 等)更为妥当些。非标准化和加速访问之间的妥协是有一定意义的。使用多个名称字段
我觉得很吃惊,许多人在数据库里就给 name 留一个字段。我觉得只有刚入门的开发人员才会这么做,但实际上网上这种做法非常普遍。我建议应该把姓氏和名字当作两个字段来处理,然后在查询的时候再把他们组合起来。

我最常用的是在同一表中创建一个计算列[字段],通过它可以自动地连接标准化后的字段,这样数据变动的时候它也跟着变。不过,这样做在采用建模软件时得很机灵才行。总之,采用连接字段的方式可以有效的隔离用户应用和开发人员界面。提防大小写混用的对象名和特殊字符
过去最令我恼火的事情之一就是数据库里有大小写混用的对象名,比如 customerdata。这一问题从 access 到 oracle 数据库都存在。我不喜欢采用这种大小写混用的对象命名方法,结果还不得不手工修改名字。想想看,这种数据库/应用程序能混到采用更强大数据库的那一天吗?采用全部大写而且包含下划符的名字具有更好的可读性(customer_data),绝对不要在对象名的字符之间留空格。小心保留词
要保证你的字段名没有和保留词、数据库系统或者常用访问方法冲突,比如,最近我编写的一个 odbc 连接程序里有个表,其中就用了 desc 作为说明字段名。后果可想而知!desc 是 descending 缩写后的保留词。表里的一个 select * 语句倒是能用,但我得到的却是一大堆毫无用处的信息。保持字段名和类型的一致性
在命名字段并为其指定数据类型的时候一定要保证一致性。假如字段在某个表中叫做“agreement_number”,你就别在另一个表里把名字改成“ref1”。假如数据类型在一个表里是整数,那在另一个表里可就别变成字符型了。记住,你干完自己的活了,其他人还要用你的数据库呢。仔细选择数字类型
在 sql 中使用 smallint 和 tinyint 类型要特别小心,比如,假如你想看看月销售总额,你的总额字段类型是 smallint,那么,如果总额超过了 $32,767 你就不能进行计算操作了。删除标记
在表中包含一个“删除标记”字段,这样就可以把行标记为删除。在关系数据库里不要单独删除某一行;最好采用清除数据程序而且要仔细维护索引整体性。避免使用触发器
触发器的功能通常可以用其他方式实现。在调试程序时触发器可能成为干扰。假如你确实需要采用触发器,你最好集中对它文档化。包含版本机制
建议你在数据库中引入版本控制机制来确定使用中的数据库的版本。无论如何你都要实现这一要求。时间一长,用户的需求总是会改变的。最终可能会要求修改数据库结构。虽然你可以通过检查新字段或者索引来确定数据库结构的版本,但我发现把版本信息直接存放到数据库中不更为方便吗?。给文本字段留足余量
id 类型的文本字段,比如客户 id 或定单号等等都应该设置得比一般想象更大,因为时间不长你多半就会因为要添加额外的字符而难堪不已。比方说,假设你的客户 id 为 10 位数长。那你应该把数据库表字段的长度设为 12 或者 13 个字符长。这算浪费空间吗?是有一点,但也没你想象的那么多:一个字段加长 3 个字符在有 1 百万条记录,再加上一点索引的情况下才不过让整个数据库多占据 3mb 的空间。但这额外占据的空间却无需将来重构整个数据库就可以实现数据库规模的增长了。身份证的号码从 15 位变成 18 位就是最好和最惨痛的例子。列[字段]命名技巧
我们发现,假如你给每个表的列[字段]名都采用统一的前缀,那么在编写 sql 表达式的时候会得到大大的简化。这样做也确实有缺点,比如破坏了自动表连接工具的作用,后者把公共列[字段]名同某些数据库联系起来,不过就连这些工具有时不也连接错误嘛。举个简单的例子,假设有两个表:
customer 和 order。customer 表的前缀是 cu_,所以该表内的子段名如下:cu_name_id、cu_surname、cu_initials 和cu_address 等。order 表的前缀是 or_,所以子段名是:
or_order_id、or_cust_name_id、or_quantity 和 or_description 等。
这样从数据库中选出全部数据的 sql 语句可以写成如下所示:
select * from customer, order where cu_surname = "myname" ;
and cu_name_id = or_cust_name_id and or_quantity = 1
在没有这些前缀的情况下则写成这个样子(用别名来区分):
select * from customer, order where customer.surname = "myname" ;
and customer.name_id = order.cust_name_id and order.quantity = 1
第 1 个 sql 语句没少键入多少字符。但如果查询涉及到 5 个表乃至更多的列[字段]你就知道这个技巧多有用了。

第 3 部分 - 选择键和索引

  1. 数据采掘要预先计划
    我所在的某一客户部门一度要处理 8 万多份联系方式,同时填写每个客户的必要数据(这绝对不是小

转载于:https://www.cnblogs.com/kelamayi/archive/2009/11/26/1611032.html

数据库设计经验谈[2]相关推荐

  1. 数据库设计经验谈之一

    一个成功的管理系统,是由:[50%的业务 + 50%的软件]所组成, 而 50%的成功软件又有[25%的数据库 + 25%的程序]所组成,数据库 设计的好坏是一个关键.如果把企业的数据比做生命所必需的 ...

  2. 数据库设计3个泛式和经验谈

    要牢记3个泛式,我把3NF缩写让大家容易记忆: 1,原子性(数据是不可以拆分的,比如1,2,3这样的就违背了1NF) 2,关联性(所有字段必须和主键关联) 3,非关联性(除主键关联外,其他字段之间不互 ...

  3. MySQL 学习笔记(14)— 数据库设计流程、实体关系图、第一范式、第二范式、第三范式、外键使用

    本文参考:https://gitbook.cn/gitchat/column/undefined/topic/5db92c12a9c3a53bc3800f0c 1. 数据库设计流程 数据库设计是对数据 ...

  4. 软件设计之 数据库设计

    [按语:在软件设计或是动态网站开发中,数据库设计时很重要,我觉得可以说是开发工作的核心部分,所以学好数据库设计,是很重要的,也是大有前途的...]  ◆.概念 首先要搞清楚容易混淆的两个概念:&quo ...

  5. 一套外企的数据库设计面试题

    最近发现园子里面关于数据库方面的文章比较多,正好我也是一个喜欢凑热闹的家伙,那就跟着烧一把火吧.^_^ 这是前阵子一个朋友面试外企的一套关于数据库设计的试题,有兴趣的朋友不妨一试. Part I   ...

  6. saas的计费数据库设计_如何构建和扩展SaaS计费解决方案

    saas的计费数据库设计 您需要的最低可行产品 (What you need for a Minimum Viable Product) When you are building your Soft ...

  7. 互联网产品mysql数据库设计总结

    mysql数据库性能不比oracle数据库,所以设计上,和oracle有一些不同.下面总结一些互联网产品的数据库设计. 1.主键 主键可以使用bigint(20) unsigned也可以使用varch ...

  8. 数据库设计 之设计 表字段类型

    2019独角兽企业重金招聘Python工程师标准>>> 数据库设计 之设计 表字段类型 博客分类: sql 之前没有 数据库设计的一些经验. 这次数据库设计.由于需求原因和没经验原因 ...

  9. 一、数据库设计与性能优化--概述

    前言 我1998年第一次接触SQL Server 6.5 for Windows NT 4.0,当时的感觉就认为SQL Server只是一个功能强大的Excel文件.现在回想起来,当年抱着这样一种态度 ...

  10. 数据库设计的10个最佳实践

    作者 | Emily Williamson 译者 | 孙薇,责编 | 屠敏 出品 | CSDN(ID:CSDNnews) 以下为译文: 数据库是应用及计算机的核心元素,负责存储运行软件应用所需的一切重 ...

最新文章

  1. MySQL-索引优化篇(3)_利用索引优化锁
  2. 【Spring注解系列13】Spring自动装配总结:@Autowired、@Resource、@Qualifier、@Inject
  3. Ubuntu16.04+ROS+ORB-SLAM2测试(转载)
  4. 【项目实战】vue+springboot项目使用富文本编辑器实现长文章发表和展示
  5. mybatis 中SQLServer 和 mysql 模糊查询 不同点
  6. 在C语言中malloc怎么声明,问下关于malloc的声明问题
  7. Linux删除所有文件(作死命令,危险命令)
  8. 数字0-9的数量(51Nod-1042)
  9. 2021年上半年系统集成项目管理工程师综合知识真题及答案解析
  10. 从.NET Core 3.1项目生成本地可执行文件
  11. android ndk 多线程mk,Android NDK 开发教程六: application.mk
  12. 为什么会出现“无法连接服务器-与网络有关或与实例有关的错误”?
  13. Hbase下载、安装流程
  14. 一位 中国70 后老程序员的 26 个职场感悟
  15. linux 的 绘画软件,Drawing Linux(简单画图工具)
  16. Qt使用dump定位崩溃位置
  17. 非接触式CPU卡的外部认证和内部认证过程
  18. html 修改表格行背景,HTML表格标记教程(20):行的背景色属性BGCOLOR
  19. java 经纬度 电子围栏_电子围栏判断 java 版本
  20. 数据结构C语言描述2(专插本/专升本)

热门文章

  1. java队列 notify_java使用线程做一个消息队列,wait,notify
  2. 计算机/程序员常用英语(持续添加,包括一些简写)
  3. 7-17 mmh学长的三色灯 (20分)
  4. 微信支付金额为0.01分报错,和少一分钱的解决办法
  5. php工作要求,PHP工作岗位要求
  6. 好程序员分享js实现简单的板球游
  7. JFinal Template Engine 使用
  8. Hive实现oracle的Minus函数
  9. Windows 7 64位下使用ADB驱动
  10. C# 导出Excel 多个Sheet