• Mysql 表的约束


    表的约束

    • 真正约束字段的是数据类型,但是数据类型约束很单一,需要有一些额外的约束,更好的保证数据的合法性,从业务逻辑角度保证数据的正确性。比如有一个字段是email,要求是唯一的。
    • 表的约束很多,这里主要介绍如下几个: null/not null,default, comment, zerofill,primary key,auto_increment,unique key

    1 空属性

    两个值:null(默认的)和not null(不为空)

    数据库默认字段基本都是字段为空,但是实际开发时,尽可能保证字段不为空,因为数据为空没办法参与运算。

    mysql> select null;
    +------+
    | NULL |
    +------+
    | NULL |
    +------+
    1 row in set (0.00 sec)
    mysql> select 1+null;
    +--------+
    | 1+null |
    +--------+
    |  NULL |
    +--------+
    1 row in set (0.00 sec)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

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

    站在正常的业务逻辑中:

    • 如果班级没有名字,你不知道你在哪个班级
    • 如果教室名字可以为空,就不知道在哪上课

    所以我们在设计数据库表的时候,一定要在表中进行限制,满足上面条件的数据就不能插入到表中。这就是“约束”。

    mysql> create table myclass(
     -> class_name varchar(20) not null,
     -> class_room varchar(10) not null);
    Query OK, 0 rows affected (0.02 sec)
    mysql> desc myclass;
    +------------+-------------+------+-----+---------+-------+
    | Field   | Type    | Null | Key | Default | Extra |
    +------------+-------------+------+-----+---------+-------+
    | class_name | varchar(20) | NO  |   | NULL  |    |
    | class_room | varchar(10) | NO  |   | NULL  |    |
    +------------+-------------+------+-----+---------+-------+
    //插入数据时,没有给教室数据插入失败:
    mysql> insert into myclass(class_name) values('class1');
    ERROR 1364 (HY000): Field 'class_room' doesn't have a default value
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    2 默认值

    默认值:某一种数据会经常性的出现某个具体的值,可以在一开始就指定好,在需要真实数据的时候,用户可以选择性的使用默认值。

    mysql> create table tt10 (
     -> name varchar(20) not null,
     -> age tinyint unsigned default 0,
     -> sex char(2) default '男'
     -> );
    Query OK, 0 rows affected (0.00 sec)
    mysql> desc tt10;
    +-------+---------------------+------+-----+---------+-------+
    | Field | Type        | Null | Key | Default | Extra |
    +-------+---------------------+------+-----+---------+-------+
    | name | varchar(20)     | NO  |   | NULL  |    |
    | age  | tinyint(3) unsigned | YES |   | 0    |    |
    | sex  | char(2)       | YES |   ||    |
    +-------+---------------------+------+-----+---------+-------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    默认值的生效:数据在插入的时候不给该字段赋值,就使用默认值

    mysql> insert into tt10(name) values('zhangsan');
    Query OK, 1 row affected (0.00 sec)
    mysql> select * from tt10;
    +----------+------+------+
    | name   | age | sex |
    +----------+------+------+
    | zhangsan |   0 ||
    +----------+------+------+
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

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

    3 列描述

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

    mysql> create table tt12 (
    -> name varchar(20) not null comment '姓名',
    -> age tinyint unsigned default 0 comment '年龄',
    -> sex char(2) default '男' comment '性别'
    -> );
    
    • 1
    • 2
    • 3
    • 4
    • 5

    –注意:not null和defalut一般不需要同时出现,因为default本身有默认值,不会为空

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

    mysql> desc tt12;
    +-------+---------------------+------+-----+---------+-------+
    | Field | Type        | Null | Key | Default | Extra |
    +-------+---------------------+------+-----+---------+-------+
    | name | varchar(20)     | NO  |   | NULL  |    |
    | age  | tinyint(3) unsigned | YES |   | 0    |    |
    | sex  | char(2)       | YES |   ||    |
    +-------+---------------------+------+-----+---------+-------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    通过show可以看到:

    mysql> show create table tt12\G
    *************************** 1. row ***************************
       Table: tt12
    Create Table: CREATE TABLE `tt12` (
    `name` varchar(20) NOT NULL COMMENT '姓名',
    `age` tinyint(3) unsigned DEFAULT '0' COMMENT '年龄',
    `sex` char(2) DEFAULT '男' COMMENT '性别'
    ) ENGINE=MyISAM DEFAULT CHARSET=gbk
    1 row in set (0.00 sec)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    4 zerofill

    刚开始学习数据库时,很多人对数字类型后面的长度很迷茫。通过show看看tt3表的建表语句

    mysql> show create table tt3\G
    ***************** 1. row *****************
    Table: tt3
    
    Create Table: CREATE TABLE `tt3` (
    	`a` int(10) unsigned DEFAULT NULL,
    	`b` int(10) unsigned DEFAULT NULL
    ) ENGINE=MyISAM DEFAULT CHARSET=gbk
    
    1 row in set (0.00 sec)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

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

    mysql> insert into tt3 values(1,2);
    Query OK, 1 row affected (0.00 sec)
    mysql> select * from tt3;
    +------+------+
    | a | b |
    +------+------+
    | 1 | 2 |
    +------+------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    但是对列添加了zerofill属性后,显示的结果就有所不同了。修改tt3表的属性:

    mysql> alter table tt3 change a a int(5) unsigned zerofill;
    mysql> show create table tt3\G
    *************************** 1. row ***************************
    Table: tt3
    Create Table: CREATE TABLE `tt3` (
    `a` int(5) unsigned zerofill DEFAULT NULL, --具有了zerofill
    `b` int(10) unsigned DEFAULT NULL
    ) ENGINE=MyISAM DEFAULT CHARSET=gbk
    1 row in set (0.00 sec)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    对a列添加了zerofill属性,再进行查找,返回如下结果:

    mysql> select * from tt3;
    +-------+------+
    | a | b |
    +-------+------+
    | 00001 | 2 |
    +-------+------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

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

    要注意的是,这只是最后显示的结果,在MySQL中实际存储的还是1。为什么是这样呢?我们可以用hex函数来证明。

    mysql> select a, hex(a) from tt3;
    +-------+--------+
    | a | hex(a) |
    +-------+--------+
    | 00001 | 1 |
    +-------+--------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    可以看出数据库内部存储的还是 1 , 00001只是设置了zerofill属性后的一种格式化输出而已。

    5 主键

    主键:primary key用来唯一的约束该字段里面的数据,不能重复,不能为空,一张表中最多只能有一个主键;主键所在的列通常是整数类型。

    案例:

    • 创建表的时候直接在字段上指定主键
    mysql> create table tt13 (
    -> id int unsigned primary key comment '学号不能为空',
    -> name varchar(20) not null);
    Query OK, 0 rows affected (0.00 sec)
    mysql> desc tt13;
    +-------+------------------+------+-----+---------+-------+
    | Field | Type | Null | Key | Default | Extra |
    +-------+------------------+------+-----+---------+-------+
    | id | int(10) unsigned | NO | PRI | NULL | | <= key 中 pri表示该字段是主键
    | name | varchar(20) | NO | | NULL | |
    +-------+------------------+------+-----+---------+-------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 主键约束:主键对应的字段中不能重复,一旦重复,操作失败。
    mysql> insert into tt13 values(1, 'aaa'); 
    Query OK, 1 row affected (0.00 sec)
    
    • 1
    • 2
    mysql> insert into tt13 values(1, 'aaa');
    ERROR 1062 (23000): Duplicate entry '1' for key 'PRIMARY'
    
    • 1
    • 2
    • 当表创建好以后但是没有主键的时候,可以再次追加主键
    alter table 表名 add primary key(字段列表)
    
    • 1
    • 删除主键
    alter table 表名 drop primary key;
    
    • 1
    mysql> alter table tt13 drop primary key;
    mysql> desc tt13;
    +-------+------------------+------+-----+---------+-------+
    | Field | Type | Null | Key | Default | Extra |
    +-------+------------------+------+-----+---------+-------+
    | id | int(10) unsigned | NO | | NULL | |
    | name | varchar(20) | NO | | NULL | |
    +-------+------------------+------+-----+---------+-------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 复合主键

    在创建表的时候,在所有字段之后,使用primary key(主键字段列表)来创建主键,如果有多个字段作为主键,可以使用复合主键。

    create table tt14(
      id int unsigned,
      course char(10) comment '课程代码',
      score tinyint unsigned default 60 comment '成绩',
      primary key(id, course) -- id和course为复合主键
      );
    
    MariaDB [t1_db]> insert into tt14 (id,course)values(2, '234');
    Query OK, 1 row affected (0.00 sec)
    
    MariaDB [t1_db]> insert into tt14 (id,course)values(2, '234');
    ERROR 1062 (23000): Duplicate entry '2-234' for key 'PRIMARY'
    
    MariaDB [t1_db]> insert into tt14 (id,course)values(1, '123');
    ERROR 1062 (23000): Duplicate entry '1-123' for key 'PRIMARY'
    
    MariaDB [t1_db]> select * from tt14;
    +----+--------+-------+
    | id | course | score |
    +----+--------+-------+
    |  1 | 123    |    60 |
    |  2 | 123    |    60 |
    |  2 | 234    |    60 |
    +----+--------+-------+
    3 rows in set (0.00 sec)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25

    注意:复合主键里,局部冲突不影响插入。

    6 自增长

    auto_increment:当对应的字段,不给值,会自动的被系统触发,系统会从当前字段中已经有的最大值+1操作,得到一个新的不同的值。通常和主键搭配使用,作为逻辑主键。

    自增长的特点:

    • 任何一个字段要做自增长,前提是本身是一个索引(key一栏有值)
    • 自增长字段必须是整数
    • 一张表最多只能有一个自增长

    案例:

    mysql> create table tt21(
    	-> id int unsigned primary key auto_increment,
    	-> name varchar(10) not null default ''
    -> );
    mysql> insert into tt21(name) values('a');
    
    mysql> insert into tt21(name) values('b');
    
    mysql> select * from tt21;
    +----+------+
    | id | name |
    +----+------+
    | 1 | a |
    | 2 | b |
    +----+------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    下面例子说明,自增长必须是整数,不能够单独使用,需要和具有唯一性的列使用。

    MariaDB [t1_db]> 
    create table tt23(
      id int unsigned  primary key auto_increment,
      name varchar(10) not null default 
      );
    Query OK, 0 rows affected (0.01 sec)
    
    MariaDB [t1_db]> 
    create table tt24(   
    id int unsigned  auto_increment,   
    name varchar(10) not null default ''    
    );
    ERROR 1075 (42000): Incorrect table definition; there can be only one auto column and it must be defined as a key
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    索引:

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

    7 唯一键

    一张表中有往往有很多字段需要唯一性,数据不能重复,但是一张表中只能有一个主键:唯一键就可以解决表中有多个字段需要唯一性约束的问题。

    唯一键的本质和主键差不多,唯一键允许为空,而且可以多个为空,空字段不做唯一性比较。关于唯一键和主键的区别:
    我们可以简单理解成,主键更多的是标识唯一性的。而唯一键更多的是保证在业务上,不要和别的信息出现重复。乍一听好像没啥区别,我们举一个例子

    假设一个场景(当然,具体可能并不是这样,仅仅为了帮助大家理解)
    比如在公司,我们需要一个员工管理系统,系统中有一个员工表,员工表中有两列信息,一个身份证号码,一个是员工工号,我们可以选择身份号码作为主键。
    
    而我们设计员工工号的时候,需要一种约束:而所有的员工工号都不能重复。
    
    具体指的是在公司的业务上不能重复,我们设计表的时候,需要这个约束,那么就可以将员工工号设计成为唯一键。
    
    一般而言,我们建议将主键设计成为和当前业务无关的字段,这样,当业务调整的时候,我们可以尽量不会对主键做过大的调整。
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    案例:

    mysql> create table student (
    -> id char(10) unique comment '学号,不能重复,但可以为空',
    -> name varchar(10)
    -> );
    Query OK, 0 rows affected (0.01 sec)
    mysql> insert into student(id, name) values('01', 'aaa');
    Query OK, 1 row affected (0.00 sec)
    
    mysql> insert into student(id, name) values('01', 'bbb'); --唯一约束不能重复
    ERROR 1062 (23000): Duplicate entry '01' for key 'id'
    mysql> insert into student(id, name) values(null, 'bbb'); -- 但可以为空
    Query OK, 1 row affected (0.00 sec)
    mysql> select * from student;
    +------+------+
    | id | name |
    +------+------+
    | 01 | aaa |
    | NULL | bbb |
    +------+------+
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    主键与外键的区别

    • 唯一键倒逼程序员插入数据时必须与自然世界贴合的,例如在自然世界里,电话和身份证号都是唯一的,那么在mysql里我们不应该只考虑主键身份证号码是唯一的,而出现重复的电话号码,因此唯一键是为了更好的约束数据与自然世界相吻合的。
    • 主键通常都是与该的数据不相关的,有了主键我们能更快的查找数据,填补了唯一键为NULL的情况。唯一键也是补充主键的不足的,当其他的列也需要唯一性或为NULL时,如果只有主键是满足不了的,其他列出现重复的值,虽然在语法上不会出错,但是在实际的业务逻辑里是有问题的。
    • 因此,主键和外键我们不应该拿来对比,而是需要根据实际业务逻辑,主键和外键互相补充的。

    8 外键

    外键用于定义主表和从表之间的关系:外键约束主要定义在从表上,主表则必须是有主键约束或unique约束。当定义外键后,要求外键列数据必须在主表的主键列存在或为null。

    外键约束的作用:

    • 外键起到表和表之间的完整性。

    外键和外键约束的区别

    语法:

    foreign key (字段名) references 主表()
    
    • 1

    案例:

    在这里插入图片描述

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

    • 先创建主键表

    create table myclass (
    id int primary key,
    name varchar(30) not null comment’班级名’
    );

    • 再创建从表
    create table stu (
    	id int primary key,
    	name varchar(30) not null comment '学生名',
    	class_id int,
    	foreign key (class_id) references myclass(id)
    );
     
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 正常插入数据
    mysql> insert into myclass values(10, 'C++大牛班'),(20, 'java大神班');
    Query OK, 2 rows affected (0.03 sec) 
    Records: 2   Duplicates: 0   Warnings: 0
    
    • 1
    • 2
    • 3
    mysql> insert into stu values(100, '张三', 10),(101, '李四',20);
    Query OK, 2 rows affected (0.01 sec) 
    Records: 2   Duplicates: 0   Warnings: 0
    
    • 1
    • 2
    • 3
    • 插入一个班级号为30的学生,因为没有这个班级,所以插入不成功
    mysql> insert into stu values(102, 'wangwu',30);
    ERROR 1452 (23000): Cannot add or update a child row:
    a foreign key constraint fails (mytest.stu, CONSTRAINT stu_ibfk_1 
    FOREIGN KEY (class_id) REFERENCES myclass (id))
    
    • 1
    • 2
    • 3
    • 4
    • 插入班级id为null,比如来了一个学生,目前还没有分配班级

    mysql> insert into stu values(102, ‘wangwu’, null);

    如何理解 外键外键约束

    • 首先我们承认,这个世界是数据很多都是相关性的。
    • 理论上,上面的例子,我们不创建外键约束,就正常建立学生表,以及班级表,该有的字段我们都有。此时,在实际使用的时候,可能会出现什么问题?
    • 有没有可能插入的学生信息中有具体的班级,但是该班级却没有在班级表中?
    • 因为此时两张表在业务上是有相关性的,但是在业务上没有建立约束关系,那么就可能出现问题。
    • 此时学生表里已经存在外键(class_id)了
    • 外键是用来实现表和表之间关系的字段,其实只是我们自己知道,但是只要是人操作,就可能犯错误。
    • 为了强约束表之间的关系,需要添加外键约束,mysql 才会帮我们维护表之间的逻辑关系。
    • 解决方案就是通过外键约束完成的。建立外键的本质其实就是把相关性交给mysql去审核了,提前告诉mysql表之间的约束关系,那么当用户插入不符合业务逻辑的数据的时候,mysql不允许你插入。

    9 综合案例 - 阅读

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

    表名字段
    商品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:

    -- 创建数据库
    create database if not exists bit32mall default character set utf8 ;
    
    -- 选择数据库
    use bit32mall;
    
    --  创建数据库表
    -- 商品
    create table if not exists goods (
    
    goods_id   int primary key auto_increment comment '商品编号', 
    goods_name varchar(32) not null comment '商品名称',
    unitprice   int   not null default 0   comment '单价,单位分',
     category   varchar(12) comment '商品分类',
    provider   varchar(64) not null comment '供应商名称'
    );
    
    -- 客户
    create table if not exists customer (
    
    customer_id   int primary key auto_increment comment '客户编号', 
    name varchar(32) not null comment '客户姓名',
    address   varchar(256)   comment '客户地址',
    email   varchar(64)   unique key comment '电子邮箱',
     sex   enum('男','女') not null comment '性别',
      card_id char(18) unique key comment '身份证'
    );
    
    -- 购买
    create table if not exists purchase (
    
    order_id   int primary key auto_increment comment '订单号',
     customer_id int comment '客户编号',
    goods_id   int   comment '商品编号',
    nums   int   default 0 comment '购买数量',
    
    foreign key (customer_id) references customer(customer_id),
    foreign key (goods_id) references goods(goods_id)
    );
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
  • 相关阅读:
    pdf压缩文件怎么压缩最小?
    微信小程序 -- 联系客服(小程序客服)
    16:00面试,16:08就出来了,问的问题有点变态。。。
    paypal订阅流程及api请求
    两天学会微服务网关Gateway-Gateway网关限流
    [MAUI]模仿哔哩哔哩的一键三连
    PC3329L DC-DC降压 10V-100V输入3A大流输出带EN功能实现零功耗只需极少元器件
    使用CAD偏移和阵列命令绘制图形、使用CAD旋转复制命令绘制图形
    数字化转型将为企业带来什么样的变革?
    C# 让程序代码在固定的线程里运行
  • 原文地址:https://blog.csdn.net/weixin_58004346/article/details/127836235