MySQL表的约束

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

文章目录

MySQL表的约束

  • 真正约束字段的是数据类型如果插入的数据超出了对应数据类型的取值范围那么数据将会插入失败。
  • 但是数据类型的约束很单一为了更好的保证数据的合法性从业务逻辑角度保证数据的正确性MySQL中出现了表的约束目的就是为了尽可能保证数据安全减少用户的误操作可能性。
  • 表的约束有很多本篇博客主要介绍如下几个null/not null、default、comment、zerofill、primary key、auto_increment、unique key、foreign key。

空属性

空属性

  • 空属性有两个值分别是null和not null。
  • 数据库默认字段基本都是允许为空的但在实际开发中我们要尽可能保证字段不为空因为空值无法参与运算。

比如通过select可以看到null的值为null。如下

在这里插入图片描述

由于空值无法参与运算因此null值加一后得到的还是null。如下

在这里插入图片描述

如果要让某个字段不允许为空在创建表的时候就可以给对应字段设置not null属性。比如我们创建一个班级表表当中包含班级名和该班级所在的教室如果插入数据时不想让这两个字段为空就可以在创建表时给这两个字段设置not null属性。如下

在这里插入图片描述

创建表完毕后查看表结构可以看到这两个字段是不允许为空的。如下

在这里插入图片描述

向表中插入记录时只有这两个字段都不为空时才能插入成功否则将会插入失败。如下

在这里插入图片描述

默认值

默认值

  • 如果某一个字段会经常性的出现某个值那么就可以考虑将这个值设置成该字段的默认值。
  • 向表中插入数据时如果不给带有默认值的字段赋值那么就会使用默认值进行插入。

比如创建一个用户表表当中包含用户的姓名、年龄和性别将用户的年龄默认设置成0将用户的性别默认设置成男。如下

在这里插入图片描述

创建表完毕后查看表结构可以看到默认值已经设置成功了。如下

在这里插入图片描述

向表中插入数据时如果不指明用户的年龄或性别那么就会使用对应的默认值如果指明了就会使用用户指定的值。如下

在这里插入图片描述

同时设置not null和default

  • 一旦给某一字段设置了默认值那么该字段将不会出现空值因为就算插入数据时没有指明该字段的值也会使用该字段的默认值进行填充。
  • 而给某一字段设置not null属性的目的是约束该字段不能为空因此一个字段设置了default属性后再设置not null属性就没有意义了。

比如创建一个id表表当中包括姓名和id将id同时设置default和not null属性。如下

在这里插入图片描述

创建表完毕后查看表结构可以看到id字段是不允许为空的并且其默认值是1。如下

在这里插入图片描述

此时在向表中插入数据时可以不指明id进行插入此时会使用id的默认值。如下

在这里插入图片描述

列描述

列描述

  • 列描述是在创建表的时候用来对各个字段进行描述的列描述会根据表创建语句保存一般是用来给程序员或DBA了解表的相关信息的相当于一种注释。

比如创建一个用户表表当中包含用户名、用户的年龄和用户的性别在每一个字段后面添加上对应的列描述。如下

在这里插入图片描述

创建表完毕后通过show create table 表名SQL可以看到创建表时的相关细节包括列描述。如下

在这里插入图片描述

zerofill

zerofill

  • 数值类型后面的圆括号中的数字代表的是显示宽度对应数值类型设置zerofill属性后如果数据的宽度小于设定的宽度则自动填充0。

比如创建一个表表当中包含a和b两列整型数据将它们的显示宽度都设置成5但是没有设置zerofill属性。如下

在这里插入图片描述

向表中插入一条记录指明a和b的值均为1由于我们没有给a和b字段设置zerofill属性因此查看表中数据时显示出来的都是1并没有显示宽度的概念。如下

在这里插入图片描述

修改表结构给a列添加上zerofill属性由于a列数据的显示宽度为5因此查看表中数据可以看到a列数据中宽度不足5位的数据都自动在前面填充0了。如下

在这里插入图片描述

需要注意的是zerofill属性的作用就是让数据以特定的方式进行显示而已数据底层的储存方式并没有发生变化通过hex函数可以看到a列中显示的00001在底层实际储存的还是1。如下

在这里插入图片描述

主键

主键

  • 我们向表当中插入一条条记录后为了方便后续进行查找我们可以选择其中的某一字段作为键值key当需要查找记录时就根据这个键值key来查找对应的记录。
  • 主键用来唯一的约束该字段里面的数据表当中每条记录的主键不能重复也不能为空并且一张表里面只能有一个主键此外主键所在的列通常是整数类型。

比如创建一个学生表表当中包含学生的学号和姓名由于学生的学号是不会重复的因此可以将其设置成主键。如下

在这里插入图片描述

创建表成功后查看表结构可以看到id对应的Key列出现了PRI这表示我们已经成功将学号设置成这张表的主键了。此外虽然在创建表的时候没有给学号设置not null属性但由于主键本身就是不能为空的因此id默认也就不能为空了。如下

在这里插入图片描述

所谓的主键约束就是插入表中的记录的主键字段不能重复如果插入记录的主键与表中已有记录的主键重复这时就会因为主键冲突而插入失败。如下

在这里插入图片描述

使用alter table 表名 drop primary keySQL即可删除指定表的主键因为一个表只有一个主键因此删除主键时只用指明要删除哪张表的主键即可。比如这里删除学生表的主键后再查看表结构可以看到id对应的Key列的PRI已经没有了。如下

在这里插入图片描述

对于已经创建好了的表使用alter table 表名 add primary key(列名)SQL可以给指定列设置成主键但是需要注意的是只有列当中的值不为空并且不重复的列才能被设置成主键。比如这里重新将学号设置成学生表的主键后再查看表结构可以看到id对应的Key列的PRI又回来了。如下

在这里插入图片描述

复合主键

  • 一张表里面只能有一个主键但一个主键可以由多个字段来承担这种主键叫做复合主键。
  • 复合主键用来唯一约束多个字段里面的数据表当中每条记录的这多个字段不能同时重复也不能为空。

比如创建一个进程表表当中包含进程的IP地址、端口号和进程的相关信息并将IP地址和端口号组合起来形成一个复合主键。如下

在这里插入图片描述

表创建完毕后查看表结构可以看到ip和port的Key列都有PRI标志并且它们都是不允许为空的。如下

在这里插入图片描述

在向进程表中插入数据时只有插入进程的IP和端口均出现冲突时才会产生主键冲突否则就允许插入。如下

在这里插入图片描述

查看表中插入的数据可以看到表当中有重复的IP地址也有重复的端口号但是不会出现IP和端口均重复的这就是复合主键的作用。如下

在这里插入图片描述

使用alter table 表名 drop primary keySQL可以删除指定表的复合主键比如这里删除进程表的复合主键后再查看表结构可以看到ip和port对应的Key列的PRI都没有了。如下

在这里插入图片描述

对于已经创建好了的表也可以使用alter table 表名 add primary key(多个列名)SQL用多个列形成复合主键但是需要注意的是被选取的多个列当中的值不能为空并且不能同时出现重复。比如这里重新将ip和port设置成进程表的复合主键后再查看表结构可以看到ip和port对应的Key列的PRI又回来了。如下

在这里插入图片描述

自增长

自增长

  • 设置了自增长属性的字段插入数据时如果不给该字段值那么系统会自动找出当前字段当中已有的最大值将最大值进行加一后的值插入该字段。
  • 任何一个字段要做自增长前提是其本身必须是一个索引Key一栏有值并且自增长字段必须是数值类型一张表最多只能有一个自增长字段。
  • 自增长通常和主键搭配使用作为逻辑主键。一般而言建议将主键设计成与当前业务无关的字段避免因为业务逻辑的调整而需要修改主键。

比如创建一个表表当中包含id和name将id同时设置成主键和自增长字段。如下

在这里插入图片描述

创建表完毕后查看表结构可以看到id的Extra列中出现了auto_increment标志。如下

在这里插入图片描述

向表中插入第一条记录时如果没有指明自增长字段的值那么自增长字段的值默认将会从1开始。如下

在这里插入图片描述

后续向表中插入记录时如果也不指明自增长字段的值那么自增长字段的值就会依次递增。如下

在这里插入图片描述

当然插入记录的时候也可以指明自增长字段的值此时将会使用该值进行插入但注意指明的值不能和表中已有的id值重复。如下

在这里插入图片描述

此后向表中插入记录时如果又不指明自增长字段的值那么自增长字段的值将会从id列中找出最大值将最大值加一后得到的值作为自增长字段的值进行插入。如下

在这里插入图片描述

说明一下 一般自增长字段设置后就不需要手动为该字段插入值了直接让其从1开始进行自增长即可。

唯一键

唯一键

  • 一张表中往往有很多字段需要唯一性但一张表中只能有一个主键而唯一键就可以解决表中有多个字段需要唯一性约束的问题。
  • 唯一键和主键都能保证字段中数据的唯一性但唯一键允许字段为空并且可以多个字段为空空字段不做唯一性比较。
  • 需要注意的是不是主键具有唯一性而是某个具有唯一性的字段被选择成为了主键而那些不是主键但是同样需要唯一性约束的字段就应该设置成唯一键。

比如创建一个学生表表当中包含学生的id、姓名和电话号码将我们选择id作为主键但同时每个学生的电话号码也应该具有唯一性约束因此应该将电话号码设置成唯一键。如下

在这里插入图片描述

表创建完毕后查看表结构可以看到tel的Key列出现了UNI标志这就表明tel已经成功被设置成唯一键了。如下

在这里插入图片描述

向表中插入记录时如果插入记录中的电话号码与表中已有记录的电话号码出现重复那么就会因为唯一键冲突而插入失败。如下

在这里插入图片描述

此外向表中插入的记录可以不指明唯一键字段的值此时该字段默认为空不做唯一性比较。如下

在这里插入图片描述

外键

外键

  • 外键用来定义主表和从表之间的关系外键约束主要定义在从表上主表必须有主键约束或唯一键约束。
  • 外键定义后要求插入外键列的数据必须在主表对应的列存在或为null。

比如先创建一个班级表作为主表表当中包含班级的id和班级名并将班级id设置为主键。如下

在这里插入图片描述

再创建一个学生表作为从表表当中包含学生的id、姓名以及学生所在班级对应的id并将学生表中的班级id列设置成外键关联到班级表中的班级id列。如下

在这里插入图片描述

表创建完毕后查看学生表的表结构可以看到学生表中的班级id对应的Key列出现了MUL标志这表明class_id已经成功被设置成了外键。如下

在这里插入图片描述

为了演示外键约束我们先向班级表中插入两条记录。如下

在这里插入图片描述

这时向学生表中插入记录时如果插入的记录对应的班级id是班级表中存在的或者插入的班级id为null那么此时是允许进行插入的。如下

在这里插入图片描述

但如果插入学生表的记录对应的班级id是3相当于插入学生表的这条记录对应的班级并不存在此时将会插入失败这就是外键约束。如下

在这里插入图片描述

这时如果向班级表中插入班级id为3的班级信息然后再向学生表中插入上述记录这时就允许插入了。如下

在这里插入图片描述

说明一下

  • 理论上来说我们创建班级表和学生表后就算不设置外键在语义上其实也已经有了外键但这样我们没办法保证后续插入学生表的记录中的班级id的正确性。
  • 而我们给学生表中的班级id设置成外键后外键约束就能保证只有班级id在班级表中存在的记录才能插入学生表否则就会插入失败。
  • 实际建立外键的本质就是把相关性交给MySQL去审核了提前告诉MySQL表之间的约束关系当用户插入不符合业务逻辑的数据时MySQL就不允许你进行插入。

综合案例

案例描述

有一个商店的数据记录客户及购物情况由以下三个表组成

  • 商品表goods商品编号goods_id商品名称goods_name单价unitprice商品类别category供应商provider。
  • 客户表customer客户编号customer_id姓名name住址address邮箱email性别sex身份证card_id。
  • 购买表purchase订单号order_id客户编号customer_id商品编号goods_id购买数量nums。

要求

  • 设置主键、外键。
  • 客户的姓名不能为空值。
  • 邮箱不能重复。
  • 客户的性别男女。

SQL编写

首先我们需要先创建一个数据库然后在该数据库中完成这三张表的创建。如下

在这里插入图片描述

创建商品表的时候将商品编号设置成主键并且可以将其设置成自增长字段其他字段的属性没有要求可以自行合理设置。如下

在这里插入图片描述

商品表创建完毕后查看表结构如下

在这里插入图片描述

创建客户表的时候将客户编号设置成主键并且可以将其设置成自增长字段然后给姓名设置not null属性将邮箱设置成唯一键将性别设置成enum类型并仅提供男女性别选项此外题目虽然没有对身份证做要求但正常来说身份证也应该保证唯一性最好也设置成唯一键。如下

在这里插入图片描述

客户表创建完毕后查看表结构如下

在这里插入图片描述

创建购买表的时候将订单号设置成主键并且可以将其设置成自增长字段然需要将客户编号和商品编号设置成外键分别关联到客户表和商品表中的客户编号和商品编号用外键约束来保证每一个订单的客户编号和商品编号都是存在的。如下

在这里插入图片描述

购买表创建完毕后查看表结构如下

在这里插入图片描述

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6
标签: mysql