【MySQL】表的约束 -- 详解

news2024/12/28 5:43:24
  • 表中一定要有各种约束,通过约束让我们在未来插入数据库表中的数据是符合预期的。约束本质是通过技术手段倒逼程序员插入正确的数据,反过来站在 MySQL 的角度,凡是插入进来的数据都是符合数据约束的。
  • 约束的最终目标:保证数据的完整性可预期性
  • 真正约束字段的是数据类型,但是数据类型约束很单一,需要有一些额外的约束,更好的保证数据的合法性,从业务逻辑角度保证数据的正确性。比如有一个字段是 email,要求是唯一的。
  • 表的约束很多,这里主要介绍如下几个: null/not null,default,comment,zerofillprimary key,auto_increment,unique key

一、非空约束

1、空属性

  • 两个值:null(默认的)和 not null不为空)
  • 数据库默认字段基本都是字段为空,但是在实际开发时,应尽可能保证字段不为空,因为数据为空没办法参与运算。

 

创建一个班级表,包含班级名和班级所在的教室。

插入数据时,没有给教室数据插入失败。 

站在正常的业务逻辑中:
如果班级没有名字,就不知道你在哪个班级;如果教室名字可以为空,就不知道你在哪上课。所以我们在设计数据库表时,一定要在表中进行限制,满足上面条件的数据就不能插入到表中。这就是 “ 约束

二、default 约束

1、默认值

默认值 :某一种数据会经常性的出现某个具体的值,可以在一开始就指定好,在需要真实数据时,用户可以选择性的使用默认值。
如果设置 default 后,用户插入了具体的数据,那么就用用户所提供的数据。
如果 数据在插入的时候不给该字段赋值,那么就使用默认值。

  • 如果没有明确指定某一列要插入,默认用的是 default。
  • 如果在建表时,对应列不能为空且默认没有设置 default 值,则无法直接插入数据。 

  • 当用户忽略某一列时,使用 default 默认值(如果设置了)。如果没有设置 default 值,则直接报错。
  • null 和 not null 主要是用来约束用户想插入的内容。
  • default 和 not null 二者并不冲突,而是相互补充的。

注意只有设置了 default 的列,才可以在插入值时,对列进行省略。

这里的 age 没有 not null 的约束,同时具有 default 默认值。

总结:

  • 如果用户想显示的插入,要么就插入一个具体的值,要么就插入 NULL。
  • 如果设置了 not null,那么插入 NULL 就会被拦截。
  • 如果没有设置 default,那么这一列就不能省略。
  • 如果在建表时没有设置 default 默认值(前提是没有设置 not null),那么 MySQL 会对我们的 sql 做优化,默认加了 default null。
  • not null 和 defalut 一般不需要同时出现,因为 default 本身有默认值,不会为空。

三、comment

1、列描述

列描述 comment ,没有实际含义,专门用来描述字段,会根据表创建语句保存,用来给程序员或  DBA  来进行了解。

通过 desc 查看不到注释信息: 

通过  show  可以看到:

四、zerofill

下面通过 show 看看 t17 表的建表语句:

可以看到 int(10),这个代表什么意思呢?整型不是 4 字节码吗?那这个 10 又代表什么呢?
其实没有  zerofill  这个属性,括号内的数字是毫无意义的。a  和  列就是前面插入的数据,如下:

但是对列添加了  zerofill  属性后,显示的结果就有所不同了。修改  t17  表的属性:
对 b   列添加了  zerofill  属性,再进行查找,返回如下结果:

这次可以看到 b 的值由原来的 2 变成 0000000002,这就是 zerofill 属性的作用,如果宽度小于设定的宽度(这里设置的是 10),自动填充 0

要注意的是,这只是最后显示的结果,在 MySQL 中实际存储的还是 2

为什么是这样呢?
我们可以用 hex(十六进制) 函数来证明。
可以看出数据库内部存储的还是 2,0000000002 只是设置了  zerofill  属性后的一种格式化输出而已。
修改括号内的值:
插入数据:
不够就补 0,补到括号内的数字位数为止,如果够了就直接显示。
为什么建表时默认是 int(10)?

因为一个整数占 4 个字节,4 个字节的取值范围:有符号是 -2^31~2^31-1,无符号是 0~2^32-1。

2^31-1 和 2^32-1 最后表示出来都是十位,所以用 int(10) 就足以把所有对应的整数的数据位表示出来。

有符号是 11 位,多了 1 是因为多了一个符号位。


五、主键

主键 primary key   用来 唯一 的约束该字段里面的数据,不能重复不能为空,一张表中最多只能有一个主键但不意味着一个表中的主键只能添加给一列。说明一个主键可以被添加到一列或多列,添加给多列则叫作复合主键。
主键所在的列通常是整数类型。

1、指定主键

  • 创建表的时候直接在字段上指定主键。

Key 中 PRI 表示该字段是主键。


2、主键约束

  • 主键约束:主键对应的字段中不能重复,一旦重复,操作失败。

3、追加主键

  • 当表创建好后但是没有主键的时候,可以再次追加主键。


4、删除主键


5、复合主键

在创建表的时候,在所有字段之后,使用  primary key(主键字段列表)   来创建主键,如果有多个字段作为主键,可以使用复合主键。
复合主键中的字段有个别发生冲突并不影响,但如果主键中多列的列信息都与历史的信息发生冲突了才会触发主键约束。

六、自增长

1、auto_increment

auto_increment: 当对应的字段不给值,会自动的被系统触发,系统会从当前字段中已经有的最大值进行  +1  操作,从而得到一个新的不同的值。
auto_increment 通常和主键搭配使用,作为逻辑主键。
自增长的特点
  • 任何一个字段要做自增长,前提是本身是一个索引key 一栏有值)。
  • 自增长字段必须是整数
  • 一张表最多只能有一个自增长

在插入后获取上次插入的 auto_increment 的值(批量插入获取的是第一个值)

索引:
  • 在关系数据库中,索引是一种单独的、物理的对数据库表中一列或多列的值进行排序的一种存储结构,它是某个表中一列或若干列值的集合和相应的指向表中物理标识这些值的数据页的逻辑指针清单。索引的作用相当于图书的目录,可以根据目录中的页码快速找到所需的内容。
  • 索引提供指向存储在表的指定列中的数据值的指针,然后根据您指定的排序顺序对这些指针排序。数据库使用索引以找到特定值,然后顺指针找到包含该值的行。这样可以使对应于表的 SQL 语句执行得更快,可快速访问数据库表中的特定信息。

七、唯一键

一张表中往往有很多字段需要唯一性,数据不能重复,但是一张表中只能有一个主键:唯一键就可以解决表中有多个字段需要唯一性约束的问题。
唯一键的本质和主键差不多,唯一键 允许为空 ,而且可以多个为空,空字段不做唯一性比较。

1、唯一键和主键的区别

可以简单理解成,主键更多的是标识唯一性的。而唯一键更多的是保证在业务上不要和别的信息出现重复。
乍一听好像没啥区别,下面给大家作一个假设来帮助理解:
在公司,我们需要一个员工管理系统,系统中有一个员工表,员工表中有两列信息,一个是身份证号码,一个是员工工号,我们可以选择身份号码作为主键。
那么我们在设计员工工号时,需要一种约束:所有的员工工号都不能重复。具体指的是在公司的业务上不能重复。
我们在设计表时,需要这个约束,那么就可以将员工工号设计成为唯一键。一般而言,建议将主键设计成和当前业务无关的字段。这样,当业务调整的时候,我们可以尽量不会对主键做过大的调整。
当 id 为 NULL 时,是可以直接插入且不冲突的。
  • 对应的主键一旦设置好,不能和其它键冲突,而且主键不能为空
  • 唯一键表达了与主键类似的功能,可以为空,NULL 不参与计算,与语言上的 NULL 不一样。

 

上面的设计中,数据库有逻辑上的错误,在现实生活中,不同的人电话号码不同。

毫无疑问,主键要保证唯一性,同时并不否认其它列属性不需要唯一性,如果也需要就得带上,避免出现上述情况。

唯一约束不能重复,但可以为 null

总结:

  • 在技术上,唯一键可以为空,主键不能为空。
  • 主键通常用来标定某一行记录在整表中的唯一性,而唯一键的侧重点在于让用户插入的列值和表中其它的列值不要发生冲突,从而保证在业务上、在上层对应的字段的唯一性。
  • 二者不是对立的,是互相补充的。主键保证表中记录的唯一性,唯一性保证列中的信息不会出现重复。

八、外键

外键 用于定义主表从表之间的关系:外键约束主要定义在从表上,主表则必须是有主键约束或  unique  约束。
当定义外键后,要求外键列数据必须在主表的主键列存在或为 null。
外键的本质:产生关联,增加约束,来整体增加表和表之间的完整性。
foreign key(字段名) references 主表(列)

对上面的示意图进行设计:

  • 先创建主键表


  • 再创建从表


  • 正常插入数据


  • 插入一个班级号为 1的学生,因为没有这个班级,所以插入不成功


  • 插入班级idnull,比如来了一个学生,目前还没有分配班级


  • 删除班级


如何理解外键约束?
首先我们承认,这个世界是数据很多都是相关性的。
理论上,上面的例子,我们不创建外键约束,就正常建立学生表和班级表,该有的字段我们都有。此时,在实际使用的时候,可能会出现什么问题?
有没有可能插入的学生信息中有具体的班级,但是该班级却没有在班级表中呢?
比如只开了 16 个 班, 但是在上课的学生里面竟然有  17  班的学生( 这个班目前并不存在) ,这很明显是有问题的。因为此时两张表在业务上是有相关性的,但是在业务上没有建立约束关系,那么就可能出现问题。
解决方案就是通过外键完成的。建立外键的本质其实就是把相关性交给 MySQL 去审核了,提前告诉 MySQL 表之间的约束关系,那么当用户插入不符合业务逻辑的数据的时候,M ySQL 不允许你插入。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1472131.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

试一下newb,还是有错误呀

解题:原式= 2. 在递增的等比数列 ( a n ) (a_n) (an​)中,若 ( a 3 − a 1 5 2 ) (a_3 - a_1 \frac{5}{2}) (a3​−a1​25​), ( a 2 3 ) (a_2 3) (a2​3), 则公比 (q) A. ( 4 3 ) ( \frac{4}{3} ) (34​) B. ( 3 2 ) ( \frac{3}{2} …

蛇形矩阵1

题目描述 把数1,2,3,…,N*N按照“蛇形1”放入N*N的矩形中,输出结果。 下面是N10的蛇形1的图示 输入格式 第一行1个正整数:N,范围在[1,100]。 输出格式 N行,每行N个整数。 输入/…

GTSAM 中的鲁棒噪声模型与 M-估计 (GTSAM Robust Noise Model and M-Estimator)

Title: GTSAM 中的鲁棒噪声模型与 M-估计 (GTSAM Robust Noise Model and M-Estimator) 文章目录 前言I. 噪声模型II. 鲁棒回归 —— M 估计1. 线性回归2. 损失函数3. 加权最小二乘 III. 鲁棒噪声模型1. 鲁棒目标函数2. 非线性加权最小二乘3. 迭代重加权最小二乘法 V. 总结参考…

第 2 章 微信小程序的构成 (代码导读)断更,后续继续更新

2.1 小程序项目的基本结构 Hello World!程序.mp4 文泉云盘 -- 图书二维码资源管理系统兆泰源二维码管理系统https://www.wqyunpan.com/resourceDetail.html?id284928&openIdoUgl9wdyNYHu9EcAe-GEwbQdZilY&qrcodeId242916&signc2lnbm1PUmNxSndPWGFOck…

rhel8配置bond遇到的问题汇总

操作环境信息 配置bond mode=4 两块网卡的配置信息 bond卡的配置信息(标红是后来加上的) 遇到的问题 rhel8 使用NetworkManage 管理网络,而不使用network,经常用到的命令是nmcli 拉起bond0后,因为没有获取到ip地址,每隔一段时间就会将网卡down掉,出现nic link is down …

代码随想录算法训练营第四三天 | 最后一块石头的重量 II、目标和、一和零

目录 最后一块石头的重量 II目标和一和零 LeetCode 1049. 最后一块石头的重量 II LeetCode 494. 目标和 LeetCode 474.一和零 最后一块石头的重量 II class Solution {// dp[j] 容量为j 的背包,最多可以背最大重量为dp[j]。// dp[j] Math.max(dp[j], dp[j - sto…

zabbix3.4.6 源码安装

Step1: 下载 https://www.zabbix.com/download 选中一下。download Zabbix Sources PackageReleaseDateRelease NotesZabbix ManualDownloadZabbix 3.4Server, Proxy, Agent, GUI3.4.615 January, 2018 Download step2 :拷贝在redhat 6.3_X86_86(192…

基于huggingface加载openai/clip-vit-large-patch14-336视觉模型demo

文章目录 引言一、模型加载二、huggingface梯度更新使用三、图像处理四、模型推理五、整体代码总结 引言 本文介绍如何使用huggingface加载视觉模型openai/clip-vit-large-patch14-336,我之所以记录此方法源于现有大模型基本采用huggingface库来加载视觉模型和大语…

【生成式AI】ChatGPT 原理解析(2/3)- 预训练 Pre-train

Hung-yi Lee 课件整理 预训练得到的模型我们叫自监督学习模型(Self-supervised Learning),也叫基石模型(foundation modle)。 文章目录 机器是怎么学习的ChatGPT里面的监督学习GPT-2GPT-3和GPT-3.5GPTChatGPT支持多语言…

nginx(二)

nginx的验证模块 输入用户名和密码 第一步先下载httpd 这个安装包 第二步编辑子配置文件 然后去网页访问192.168.68.3/admin/ 连接之后,会出现404,404出现是因为没给网页写页面 如果要写页面,则在/opt/html,建立一个admin&#x…

算法沉淀——动态规划之路径问题(leetcode真题剖析)

算法沉淀——动态规划之路径问题 01.不同路径02.不同路径 II03.珠宝的最高价值04.下降路径最小和05.最小路径和06.地下城游戏 01.不同路径 题目链接:https://leetcode.cn/problems/unique-paths/ 一个机器人位于一个 m x n 网格的左上角 (起始点在下图…

探索C语言位段的秘密

位段 1. 什么是位段2. 位段的内存分配3. 位段的跨平台问题4. 位段的应用4. 使用位段的注意事项 1. 什么是位段 我们使用结构体实现位段,位段的声明和结构体是类似的,有两个不同: 位段的成员必须是int,unsigned int,或…

IO进程线程作业day7

信号灯集共享内存 自定义头文件 #ifndef SEM_H_ #define SEM_H_ //创建信号灯集, int creat_t(int number); //申请释放资源 int P(int semid,int semno); //申请释放资源 int V(int semid,int semno); //删除信号灯集 int del(int semid); #endif信号灯集函数集合 #include…

多维时序 | Matlab实现GRU-MATT门控循环单元融合多头注意力多变量时间序列预测模型

多维时序 | Matlab实现GRU-MATT门控循环单元融合多头注意力多变量时间序列预测模型 目录 多维时序 | Matlab实现GRU-MATT门控循环单元融合多头注意力多变量时间序列预测模型预测效果基本介绍程序设计参考资料 预测效果 基本介绍 1.多维时序 | Matlab实现GRU-MATT门控循环单元融…

wcf 简单实践 数据绑定 数据更新ui

1.概要 2.代码 2.1 xaml <Window x:Class"WpfApp3.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d"http://schemas.microsoft.com/expr…

《数据治理简易速速上手小册》第2章 数据治理框架的建立(2024 最新版)

文章目录 2.1 确立数据治理目标2.1.1 基础知识2.1.2 重点案例&#xff1a;提升零售业数据质量2.1.3 拓展案例 1&#xff1a;银行的数据安全加强2.1.4 拓展案例 2&#xff1a;医疗机构的合规性数据报告 2.2 数据治理框架的关键要素2.2.1 基础知识2.2.2 重点案例&#xff1a;制药…

VM-UNet:视觉Mamba UNet用来医学图像分割 论文及代码解读

VM-UNet 期刊分析摘要贡献方法整体框架1. Vision Mamba UNet (VM-UNet)2. VSS block3. Loss function 实验1.对比实验2.消融实验 可借鉴参考代码使用 期刊分析 期刊名&#xff1a; Arxiv 论文主页&#xff1a; VM-UNet 代码&#xff1a; Code 摘要 在医学图像分割领域&#x…

JANGOW: 1.0.1

kali:192.168.223.128 主机发现 nmap -sP 192.168.223.0/24 端口扫描 nmap -p- 192.168.223.154 开启了21 80端口 web看一下&#xff0c;有个busque.php参数是buscar,但是不知道输入什么&#xff0c;尝试文件包含失败 扫描目录 dirsearch -u http://192.168.223.154 dirse…

mysql 分表实战

本文主要介绍基于range分区的相关 1、业务需求&#xff0c;每日160w数据&#xff0c;每月2000w;解决大表数据读写性能问题。 2、数据库mysql 8.0.34&#xff0c;默认innerDB;mysql自带的逻辑分表 3、分表的目的:解决大表性能差&#xff0c;小表缩小查询单位的特点(其实优化的精…

人事|人事管理系统|基于Springboot的人事管理系统设计与实现(源码+数据库+文档)

人事管理系统目录 目录 基于Springboot的人事管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、管理员登录 2、员工管理 3、公告信息管理 4、公告类型管理 5、培训管理 6、培训类型管理 四、数据库设计 1、实体ER图 五、核心代码 六、论文参考 七、…