Mysql 完整性约束

Posted 爱学习爱生活的小九

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Mysql 完整性约束相关的知识,希望对你有一定的参考价值。

约束条件与数据类型的宽度一样,都是可选参数

作用:用于保证数据的完整性和一致性
主要分为:

PRIMARY KEY (PK)    标识该字段为该表的主键,可以唯一的标识记录
FOREIGN KEY (FK)    标识该字段为该表的外键
NOT NULL    标识该字段不能为空
UNIQUE KEY (UK)    标识该字段的值是唯一的
AUTO_INCREMENT    标识该字段的值自动增长(整数类型,而且为主键)
DEFAULT    为该字段设置默认值

UNSIGNED 无符号
ZEROFILL 使用0填充

说明:

1. 是否允许为空,默认NULL,可设置NOT NULL,字段不允许为空,必须赋值
2. 字段是否有默认值,缺省的默认值是NULL,如果插入记录时不给字段赋值,此字段使用默认值
sex enum(‘male‘,‘female‘) not null default ‘male‘
age int unsigned NOT NULL default 20 必须为正值(无符号) 不允许为空 默认是20
3. 是否是key
主键 primary key
外键 foreign key
索引 (index,unique...)

二、not null与default

是否可空,null表示空,非字符串
not null - 不可空
null - 可空

默认值,创建列时可以指定默认值,当插入数据时如果未主动设置,则自动添加默认值

 

 

create table t1 (
id int not null  default 2  ,
num int not null
);


MariaDB [task]> desc t4
    -> ;
+-------+---------+------+-----+---------+-------+
| Field | Type    | Null | Key | Default | Extra |
+-------+---------+------+-----+---------+-------+
| id    | int(11) | NO   |     | 2       |       |
| num   | int(11) | NO   |     | NULL    |       |
+-------+---------+------+-----+---------+-------+
==================default====================
#设置id字段有默认值后,则无论id字段是null还是not null,都可以插入空,插入空默认填入default指定的默认值


三、 unique

============设置唯一约束 UNIQUE===============
方法一

create table t5 (
id int ,
name varchar(50) unique #该值不可重复
);

方法二
MariaDB [task]> create table t6 (
    -> id int,
    -> name  varchar(50),
    -> constraint test_name  unique(name)  ,约束,名称, 方法(属性)
    -> );


MariaDB [task]> insert into t5 values  (1,小九);
Query OK, 1 row affected (0.00 sec)

MariaDB [task]> insert into t5 values  (1,小九);
ERROR 1062 (23000): Duplicate entry 小九 for key name
MariaDB [task]> 

 

 

联合唯一,单一记录可以重复,多条记录不可重复,  如 IP可以重复,端口也可以重复,但是一起就不能重复,

create table service(
id int primary key auto_increment,
name varchar(20),
host varchar(15) not null,
port int not null,
unique(host,port) #联合唯一
);


MariaDB [task]> desc service ;
+-------+-------------+------+-----+---------+----------------+
| Field | Type        | Null | Key | Default | Extra          |
+-------+-------------+------+-----+---------+----------------+
| id    | int(11)     | NO   | PRI | NULL    | auto_increment |
| name  | varchar(20) | YES  |     | NULL    |                |
| host  | varchar(15) | NO   | MUL | NULL    |                |
| port  | int(11)     | NO   |     | NULL    |                |

 

四、 primary key

primary key字段的值不为空且唯一

一个表中可以:

单列做主键
多列做主键(复合主键)

但一个表内只能有一个主键primary key

这里不唯一切空白,大家肯定想到了  not null  + unique ,其实 primary key 就是这两种方法的结合

一、单列主键

MariaDB [task]> create table t7 (
 id int primary key, #这里的 primary key 可以替换为 not null unique 
 name varchar(50) not null 
);
Query OK, 0 rows affected (0.00 sec)

MariaDB [task]> desc t7;
+-------+-------------+------+-----+---------+-------+
| Field | Type        | Null | Key | Default | Extra |
+-------+-------------+------+-----+---------+-------+
| id    | int(11)     | NO   | PRI | NULL    |       |
| name  | varchar(50) | NO   |     | NULL    |       |
+-------+-------------+------+-----+---------+-------+

方法二、
create table t8(
id int,
name varchar(20),
comment varchar(100),
constraint pk_name primary key(id); #创建主键并为其命名pk_name

 

 多列主键

MariaDB [task]> create table t8 (
    -> ip varchar(15),
    -> port char(5),
    -> service_name varchar(10) not null ,
    -> primary key(ip,port) # IP 可以重复,端口可以重复, 但是IP+端口不可重复。
    -> );
Query OK, 0 rows affected (0.01 sec)

MariaDB [task]> 
MariaDB [task]> desc t8 ;
+--------------+-------------+------+-----+---------+-------+
| Field        | Type        | Null | Key | Default | Extra |
+--------------+-------------+------+-----+---------+-------+
| ip           | varchar(15) | NO   | PRI |         |       |
| port         | char(5)     | NO   | PRI |         |       |
| service_name | varchar(10) | NO   |     | NULL    |       |
+--------------+-------------+------+-----+---------+-------+
3 rows in set (0.00 sec)

MariaDB [task]> insert into t8 values
    -> (127.0.0.1,80,httpd),
    -> (127.0.0.2,80,apache)
    -> ;
Query OK, 2 rows affected (0.00 sec)
Records: 2  Duplicates: 0  Warnings: 0

MariaDB [task]> insert into t8 values
    -> (127.0.0.1,80,httpd);
ERROR 1062 (23000): Duplicate entry 127.0.0.1-80 for key PRIMARY
MariaDB [task]> 

 

五、 auto_increment

约束字段为自动增长,被约束的字段必须同时被key约束

MariaDB [task]> insert into t8 values
    -> (127.0.0.1,80,httpd);
ERROR 1062 (23000): Duplicate entry 127.0.0.1-80 for key PRIMARY
MariaDB [task]> create  table t9 (
    -> id int primary key   auto_increment,
    -> name  varchar(10)
    -> );


MariaDB [task]> insert into t9(name)  values 
    -> (xiaojiu) ;

MariaDB [task]> insert into t9(name)  values 
    -> (xiao) ;

MariaDB [task]> select *from t9 ;
+----+---------+
| id | name    |
+----+---------+
|  1 | xiaojiu |
|  2 | xiao    |
+----+---------+

查看步长

 

MariaDB [(none)]> show variables like auto_incre%;
+--------------------------+-------+
| Variable_name            | Value |
+--------------------------+-------+
| auto_increment_increment | 1     |
| auto_increment_offset    | 1     |
+--------------------------+-------+


设置步长
#设置步长
sqlserver:自增步长
    基于表级别
    create table t1(
        id int。。。
    )engine=innodb,auto_increment=2 步长=2 default charset=utf8

mysql自增的步长:
    show session variables like auto_inc%;

    #基于会话级别
    set session auth_increment_increment=2 #修改会话级别的步长

    #基于全局级别的
    set global auth_increment_increment=2 #修改全局级别的步长(所有会话都生效)


#!!!注意了注意了注意了!!!
If the value of auto_increment_offset is greater than that of auto_increment_increment, the value of auto_increment_offset is ignored. 
翻译:如果auto_increment_offset的值大于auto_increment_increment的值,则auto_increment_offset的值会被忽略 
比如:设置auto_increment_offset=3,auto_increment_increment=2

 

六、 foreign key  外键

一 快速理解foreign key

员工信息表有三个字段:工号 姓名 部门

公司有3个部门,但是有1个亿的员工,那意味着部门这个字段需要重复存储,部门名字越长,越浪费

解决方法:

我们完全可以定义一个部门表

然后让员工信息表关联该表,如何关联,即foreign key

#表类型必须是innodb存储引擎,且被关联的字段,即references指定的另外一个表的字段,必须保证唯一
 
create table t11 (  #创建父表
id int primary key ,
name varchar(50) not null
);

create table t12(
id int primary key,
name varchar(10),
t10_id int,
constraint  t11_name  foreign key(t10_id) #关联字段
references  t12(id)  #关联父表名称及字段
on delete cascade  #绑定关系,删除父表字段 子表会跟着删除
on update cascade # 绑定关系, 更新父表会跟着更新
)engine=innodb; #指定 表类型 innodb

insert into  t11 values
(1,欧德博爱技术有限事业部),
(2,艾利克斯人力资源部),
(3,销售部);

insert into t12 values
(1,hha,1),
(2,hehe,2),
(3,gaga,2)
;

二 如何找出两张表之间的关系

分析步骤:
#1、先站在左表的角度去找
是否左表的多条记录可以对应右表的一条记录,如果是,则证明左表的一个字段foreign key 右表一个字段(通常是id)

#2、再站在右表的角度去找
是否右表的多条记录可以对应左表的一条记录,如果是,则证明右表的一个字段foreign key 左表一个字段(通常是id)

#3、总结:
#多对一:
如果只有步骤1成立,则是左表多对一右表
如果只有步骤2成立,则是右表多对一左表

#多对多
如果步骤1和2同时成立,则证明这两张表时一个双向的多对一,即多对多,需要定义一个这两张表的关系表来专门存放二者的关系

#一对一:
如果1和2都不成立,而是左表的一条记录唯一对应右表的一条记录,反之亦然。这种情况很简单,就是在左表foreign key右表的基础上,将左表的外键字段设置成unique即可




 














以上是关于Mysql 完整性约束的主要内容,如果未能解决你的问题,请参考以下文章

一篇文章带你彻底了解MySQL各种约束

MySQL的约束

十四MySQL 约束详解

MySQL表的完整性约束

MySQL表的完整性约束

mysql表的完整性约束