#第13章_约束

/*

  1. 基础知识
    1.1 为什么需要约束? 为了保证数据的完整性!

1.2 什么叫约束?对表中字段的限制。

1.3 约束的分类:

角度1:约束的字段的个数
单列约束 vs 多列约束

角度2:约束的作用范围

列级约束:将此约束声明在对应字段的后面
表级约束:在表中所有字段都声明完,在所有字段的后面声明的约束

角度3:约束的作用(或功能)

① not null (非空约束)
② unique (唯一性约束)
③ primary key (主键约束)
④ foreign key (外键约束)
⑤ check (检查约束)
⑥ default (默认值约束)

1.4 如何添加/删除约束?

CREATE TABLE时添加约束

ALTER TABLE 时增加约束、删除约束

*/

#2. 如何查看表中的约束
SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘test1’;

CREATE DATABASE dbtest13;
USE dbtest13;
#3. not null (非空约束)
#3.1 在CREATE TABLE时添加约束

CREATE TABLE test1(
id INT NOT NULL,
last_name VARCHAR(15) NOT NULL,
email VARCHAR(25),
salary DECIMAL(10,2)

);

DESC test1;

INSERT INTO test1(id,last_name,email,salary)
VALUES(1,‘Tom’,‘tom@126.com’,3400);

#错误:Column ‘last_name’ cannot be null
INSERT INTO test1(id,last_name,email,salary)
VALUES(2,NULL,‘tom1@126.com’,3400);

#错误:Column ‘id’ cannot be null
INSERT INTO test1(id,last_name,email,salary)
VALUES(NULL,‘Jerry’,‘jerry@126.com’,3400);

INSERT INTO test1(id,email)
VALUES(2,‘abc@126.com’);

UPDATE test1
SET last_name = NULL
WHERE id = 1;

UPDATE test1
SET email = ‘tom@126.com’
WHERE id = 1;

#3.2 在ALTER TABLE时添加约束
SELECT * FROM test1;

DESC test1;

ALTER TABLE test1
MODIFY email VARCHAR(25) NOT NULL;

#3.3 在ALTER TABLE时删除约束
ALTER TABLE test1
MODIFY email VARCHAR(25) NULL;

#4. unique (唯一性约束)

#4.1 在CREATE TABLE时添加约束
CREATE TABLE test2(
id INT UNIQUE, #列级约束
last_name VARCHAR(15) ,
email VARCHAR(25),
salary DECIMAL(10,2),
#表级约束
CONSTRAINT uk_test2_email UNIQUE(email)
);

DESC test2;

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘test2’;

#在创建唯一约束的时候,如果不给唯一约束命名,就默认和列名相同。

INSERT INTO test2(id,last_name,email,salary)
VALUES(1,‘Tom’,‘tom@126.com’,4500);

#错误:Duplicate entry ‘1’ for key ‘test2.id’
INSERT INTO test2(id,last_name,email,salary)
VALUES(1,‘Tom1’,‘tom1@126.com’,4600);

#错误:Duplicate entry ‘tom@126.com’ for key ‘test2.uk_test2_email’
INSERT INTO test2(id,last_name,email,salary)
VALUES(2,‘Tom1’,‘tom@126.com’,4600);

#可以向声明为unique的字段上添加null值。而且可以多次添加null
INSERT INTO test2(id,last_name,email,salary)
VALUES(2,‘Tom1’,NULL,4600);

INSERT INTO test2(id,last_name,email,salary)
VALUES(3,‘Tom2’,NULL,4600);

SELECT * FROM test2;

#4.2 在ALTER TABLE时添加约束

DESC test2;

UPDATE test2
SET salary = 5000
WHERE id = 3;
#方式1:
ALTER TABLE test2
ADD CONSTRAINT uk_test2_sal UNIQUE(salary);
#方式2:
ALTER TABLE test2
MODIFY last_name VARCHAR(15) UNIQUE;

#4.3 复合的唯一性约束
CREATE TABLE USER(
id INT,
name VARCHAR(15),
password VARCHAR(25),

#表级约束
CONSTRAINT uk_user_name_pwd UNIQUE(name,password)
);

INSERT INTO USER
VALUES(1,‘Tom’,‘abc’);
#可以成功的:
INSERT INTO USER
VALUES(1,‘Tom1’,‘abc’);

SELECT *
FROM USER;

#案例:复合的唯一性约束的案例
#学生表
CREATE TABLE student(
sid INT, #学号
sname VARCHAR(20), #姓名
tel CHAR(11) UNIQUE KEY, #电话
cardid CHAR(18) UNIQUE KEY #身份证号
);

#课程表
CREATE TABLE course(
cid INT, #课程编号
cname VARCHAR(20) #课程名称
);

#选课表
CREATE TABLE student_course(
id INT,
sid INT, #学号
cid INT, #课程编号
score INT,
UNIQUE KEY(sid,cid) #复合唯一
);
INSERT INTO student VALUES(1,‘张三’,‘13710011002’,‘101223199012015623’);#成功
INSERT INTO student VALUES(2,‘李四’,‘13710011003’,‘101223199012015624’);#成功
INSERT INTO course VALUES(1001,‘Java’),(1002,‘MySQL’);#成功

SELECT * FROM student;

SELECT * FROM course;

INSERT INTO student_course VALUES
(1, 1, 1001, 89),
(2, 1, 1002, 90),
(3, 2, 1001, 88),
(4, 2, 1002, 56);#成功

SELECT * FROM student_course;

#错误:Duplicate entry ‘2-1002’ for key ‘student_course.sid’
INSERT INTO student_course VALUES
(5,2,1002,67);

#4.4 删除唯一性约束
– 添加唯一性约束的列上也会自动创建唯一索引。
– 删除唯一约束只能通过删除唯一索引的方式删除。
– 删除时需要指定唯一索引名,唯一索引名就和唯一约束名一样。
– 如果创建唯一约束时未指定名称,如果是单列,就默认和列名相同;如果是组合列,那么默认和()中排在第一个的列名相同。也可以自定义唯一性约束名。

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘student_course’;

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘test2’;

DESC test2;

#如何删除唯一性索引
ALTER TABLE test2
DROP INDEX last_name;

ALTER TABLE test2
DROP INDEX uk_test2_sal;

#5. primary key (主键约束)
#5.1 在CREATE TABLE时添加约束

#一个表中最多只能有一个主键约束。

#错误:Multiple primary key defined
CREATE TABLE test3(
id INT PRIMARY KEY, #列级约束
last_name VARCHAR(15) PRIMARY KEY,
salary DECIMAL(10,2),
email VARCHAR(25)
);

主键约束特征:非空且唯一,用于唯一的标识表中的一条记录。

CREATE TABLE test4(
id INT PRIMARY KEY, #列级约束
last_name VARCHAR(15),
salary DECIMAL(10,2),
email VARCHAR(25)
);

#MySQL的主键名总是PRIMARY,就算自己命名了主键约束名也没用。
CREATE TABLE test5(
id INT ,
last_name VARCHAR(15),
salary DECIMAL(10,2),
email VARCHAR(25),
#表级约束
CONSTRAINT pk_test5_id PRIMARY KEY(id) #没有必要起名字。
);

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘test5’;

INSERT INTO test4(id,last_name,salary,email)
VALUES(1,‘Tom’,4500,‘tom@126.com’);

#错误:Duplicate entry ‘1’ for key ‘test4.PRIMARY’
INSERT INTO test4(id,last_name,salary,email)
VALUES(1,‘Tom’,4500,‘tom@126.com’);

#错误:Column ‘id’ cannot be null
INSERT INTO test4(id,last_name,salary,email)
VALUES(NULL,‘Tom’,4500,‘tom@126.com’);

SELECT * FROM test4;

CREATE TABLE user1(
id INT,
NAME VARCHAR(15),
PASSWORD VARCHAR(25),

PRIMARY KEY (NAME,PASSWORD)

);
#如果是多列组合的复合主键约束,那么这些列都不允许为空值,并且组合的值不允许重复。
INSERT INTO user1
VALUES(1,‘Tom’,‘abc’);

INSERT INTO user1
VALUES(1,‘Tom1’,‘abc’);
#错误:Column ‘name’ cannot be null
INSERT INTO user1
VALUES(1,NULL,‘abc’);

SELECT * FROM user1;

#5.2 在ALTER TABLE时添加约束

CREATE TABLE test6(
id INT ,
last_name VARCHAR(15),
salary DECIMAL(10,2),
email VARCHAR(25)
);

DESC test6;

ALTER TABLE test6
ADD PRIMARY KEY (id);

#5.3 如何删除主键约束 (在实际开发中,不会去删除表中的主键约束!)
ALTER TABLE test6
DROP PRIMARY KEY;

#6. 自增长列:AUTO_INCREMENT

6.1 在CREATE TABLE时添加

CREATE TABLE test7(
id INT PRIMARY KEY AUTO_INCREMENT,
last_name VARCHAR(15)
);
#开发中,一旦主键作用的字段上声明有AUTO_INCREMENT,则我们在添加数据时,就不要给主键
#对应的字段去赋值了。
INSERT INTO test7(last_name)
VALUES(‘Tom’);

SELECT * FROM test7;

#当我们向主键(含AUTO_INCREMENT)的字段上添加0 或 null时,实际上会自动的往上添加指定的字段的数值
INSERT INTO test7(id,last_name)
VALUES(0,‘Tom’);

INSERT INTO test7(id,last_name)
VALUES(NULL,‘Tom’);

INSERT INTO test7(id,last_name)
VALUES(10,‘Tom’);

INSERT INTO test7(id,last_name)
VALUES(-10,‘Tom’);

#6.2 在ALTER TABLE 时添加
CREATE TABLE test8(
id INT PRIMARY KEY ,
last_name VARCHAR(15)
);

DESC test8;

ALTER TABLE test8
MODIFY id INT AUTO_INCREMENT;

#6.3 在ALTER TABLE 时删除

ALTER TABLE test8
MODIFY id INT ;

#6.4 MySQL 8.0新特性—自增变量的持久化

#在MySQL 5.7中演示
CREATE TABLE test9(
id INT PRIMARY KEY AUTO_INCREMENT
);

INSERT INTO test9
VALUES(0),(0),(0),(0);

SELECT * FROM test9;

DELETE FROM test9
WHERE id = 4;

INSERT INTO test9
VALUES(0);

DELETE FROM test9
WHERE id = 5;

#重启服务器

SELECT * FROM test9;

INSERT INTO test9
VALUES(0);

#在MySQL 8.0中演示
CREATE TABLE test9(
id INT PRIMARY KEY AUTO_INCREMENT
);

INSERT INTO test9
VALUES(0),(0),(0),(0);

SELECT * FROM test9;

DELETE FROM test9
WHERE id = 4;

INSERT INTO test9
VALUES(0);

DELETE FROM test9
WHERE id = 5;

#重启服务器

SELECT * FROM test9;

INSERT INTO test9
VALUES(0);

#7.foreign key (外键约束)
#7.1 在CREATE TABLE 时添加

#主表和从表;父表和子表

#①先创建主表
CREATE TABLE dept1(
dept_id INT,
dept_name VARCHAR(15)
);
#②再创建从表
CREATE TABLE emp1(
emp_id INT PRIMARY KEY AUTO_INCREMENT,
emp_name VARCHAR(15),
department_id INT,

#表级约束
CONSTRAINT fk_emp1_dept_id FOREIGN KEY (department_id) REFERENCES dept1(dept_id)

);

#上述操作报错,因为主表中的dept_id上没有主键约束或唯一性约束。
#③ 添加
ALTER TABLE dept1
ADD PRIMARY KEY (dept_id);

DESC dept1;

#④ 再创建从表
CREATE TABLE emp1(
emp_id INT PRIMARY KEY AUTO_INCREMENT,
emp_name VARCHAR(15),
department_id INT,

#表级约束
CONSTRAINT fk_emp1_dept_id FOREIGN KEY (department_id) REFERENCES dept1(dept_id)

);

DESC emp1;

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘emp1’;

#7.2 演示外键的效果
#添加失败
INSERT INTO emp1
VALUES(1001,‘Tom’,10);

INSERT INTO dept1
VALUES(10,‘IT’);
#在主表dept1中添加了10号部门以后,我们就可以在从表中添加10号部门的员工
INSERT INTO emp1
VALUES(1001,‘Tom’,10);

#删除失败
DELETE FROM dept1
WHERE dept_id = 10;

#更新失败
UPDATE dept1
SET dept_id = 20
WHERE dept_id = 10;

#7.3 在ALTER TABLE时添加外键约束
CREATE TABLE dept2(
dept_id INT PRIMARY KEY,
dept_name VARCHAR(15)
);

CREATE TABLE emp2(
emp_id INT PRIMARY KEY AUTO_INCREMENT,
emp_name VARCHAR(15),
department_id INT
);

ALTER TABLE emp2
ADD CONSTRAINT fk_emp2_dept_id FOREIGN KEY(department_id) REFERENCES dept2(dept_id);

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘emp2’;

#7.4 ### 约束等级

Cascade方式:在父表上update/delete记录时,同步update/delete掉子表的匹配记录

Set null方式:在父表上update/delete记录时,将子表上匹配记录的列设为null,但是要注意子表的外键列不能为not null

No action方式:如果子表中有匹配的记录,则不允许对父表对应候选键进行update/delete操作

Restrict方式:同no action, 都是立即检查外键约束

Set default方式(在可视化工具SQLyog中可能显示空白):父表有变更时,子表将外键列设置成一个默认的值,但Innodb不能识别

#演示:

on update cascade on delete set null

CREATE TABLE dept(
did INT PRIMARY KEY, #部门编号
dname VARCHAR(50) #部门名称
);

CREATE TABLE emp(
eid INT PRIMARY KEY, #员工编号
ename VARCHAR(5), #员工姓名
deptid INT, #员工所在的部门
FOREIGN KEY (deptid) REFERENCES dept(did) ON UPDATE CASCADE ON DELETE SET NULL
#把修改操作设置为级联修改等级,把删除操作设置为set null等级
);

INSERT INTO dept VALUES(1001,‘教学部’);
INSERT INTO dept VALUES(1002, ‘财务部’);
INSERT INTO dept VALUES(1003, ‘咨询部’);

INSERT INTO emp VALUES(1,‘张三’,1001); #在添加这条记录时,要求部门表有1001部门
INSERT INTO emp VALUES(2,‘李四’,1001);
INSERT INTO emp VALUES(3,‘王五’,1002);

UPDATE dept
SET did = 1004
WHERE did = 1002;

DELETE FROM dept
WHERE did = 1004;

SELECT * FROM dept;

SELECT * FROM emp;

#结论:对于外键约束,最好是采用: ON UPDATE CASCADE ON DELETE RESTRICT 的方式。

#7.5 删除外键约束

#一个表中可以声明有多个外键约束
USE atguigudb;
SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘employees’;

USE dbtest13;

SELECT * FROM information_schema.table_constraints
WHERE table_name = ‘emp1’;

#删除外键约束

ALTER TABLE emp1
DROP FOREIGN KEY fk_emp1_dept_id;

#再手动的删除外键约束对应的普通索引
SHOW INDEX FROM emp1;

ALTER TABLE emp1
DROP INDEX fk_emp1_dept_id;

#8. check 约束

MySQL5.7 不支持CHECK约束,MySQL8.0支持CHECK约束。

CREATE TABLE test10(
id INT,
last_name VARCHAR(15),
salary DECIMAL(10,2) CHECK(salary > 2000)
);

INSERT INTO test10
VALUES(1,‘Tom’,2500);

#添加失败
INSERT INTO test10
VALUES(2,‘Tom1’,1500);

SELECT * FROM test10;

#9.DEFAULT约束
#9.1 在CREATE TABLE添加约束
CREATE TABLE test11(
id INT,
last_name VARCHAR(15),
salary DECIMAL(10,2) DEFAULT 2000
);

DESC test11;

INSERT INTO test11(id,last_name,salary)
VALUES(1,‘Tom’,3000);

INSERT INTO test11(id,last_name)
VALUES(2,‘Tom1’);

SELECT *
FROM test11;

#9.2 在ALTER TABLE添加约束
CREATE TABLE test12(
id INT,
last_name VARCHAR(15),
salary DECIMAL(10,2)
);

DESC test12;

ALTER TABLE test12
MODIFY salary DECIMAL(8,2) DEFAULT 2500;

#9.3 在ALTER TABLE删除约束
ALTER TABLE test12
MODIFY salary DECIMAL(8,2);

SHOW CREATE TABLE test12;

Mysql 约束练习【第13章_约束】相关推荐

  1. MySQL_第13章_约束

    第13章_约束 1. 约束(constraint)概述 1.1 为什么需要约束 数据完整性(Data Integrity)是指数据的精确性(Accuracy)和可靠性(Reliability).它是防 ...

  2. c语言程序设计实验13文件,第13章_文件---《C语言程序设计》实验指导.ppt

    第13章_文件---<C语言程序设计>实验指导 第十三章 主要内容 13.1 C文件概述 13.2 文件类型指针 13.3 文件的打开与关闭 13.4 文件的读写 13.5 文件的定位 1 ...

  3. MySQL基础篇-第14章_视图

    第14章_视图 讲师:尚硅谷-宋红康(江湖人称:康师傅) 官网:http://www.atguigu.com 1. 常见的数据库对象 对象 描述 表(TABLE) 表是存储数据的逻辑单元,以行和列的形 ...

  4. MySQL笔记:第14章_视图

    第14章_视图 1. 常见的数据库对象 2. 视图概述 2.1 为什么使用视图? 2.2 视图的理解 演示 3. 创建视图 3.1 创建单表视图 3.2 创建多表联合视图 3.3 基于视图创建视图 演 ...

  5. MySQL笔记:第11章_数据库的设计规范

    第11章_数据库的设计规范 1. 为什么需要数据库设计 2.范式 2.1范式简介 2.2范式都包括哪些 2.3 键和相关属性的概念 2.4第一范式(1st NF) 2.5 第二范式(2nd NF) 2 ...

  6. MySQL笔记:第06章_多表查询

    第06章_多表查询 多表查询概述 1.一个案例引发的多表连接 1.1 案例说明 1.2 笛卡尔积(或交叉连接)的理解 演示代码 2. 多表查询分类讲解 分类1:等值连接 vs 非等值连接 等值连接 非 ...

  7. MySQL笔记:第05章_存储引擎

    第05章_存储引擎 1. 查看存储引擎 2. 设置系统默认的存储引擎 3. 设置表的存储引擎 3.1 创建表时指定存储引擎 3.2 修改表的存储引擎 4. 引擎介绍 4.1 InnoDB 引擎:具备外 ...

  8. 明解C语言入门篇_第13章_文件处理

    前言 本文为业余学习<明解C语言入门篇>的记录,包含代码清单和练习题. 开始学习时间:2022年8月21日 +++++++++++++++++++++++++++++++ 第1章 初识C语 ...

  9. masonry的约束应该写在哪里_约束选择

    本文用WarpDrive作为例子讨论一下模块设计的思路.这种问题是一线软件设计工程师在设计中天天面对的问题,但这个东西很难告诉你:第一步怎么做,第二步怎么做.所以你也别来找我要什么"in n ...

最新文章

  1. 2月07日云栖精选夜读:观点 | 阿里云 MVP 唐俊飞:安全性可以认为是一种能力...
  2. Java的Scanner类的next与nextLine用法区别
  3. linux中UDP程序流程、客户端、服务端
  4. 控制台启动jar包关闭控制台怎么关java程序_Jpom一款简而轻的低侵入式Java运维、监控软件...
  5. 休眠 嵌入式_内幕消息:嵌入式软件挤出最低功耗模式
  6. STM8单片机ADC单次采样模式
  7. BPMN 2.0 流程设计
  8. MFC鼠标单击截获鼠标双击事件,且无法记录单击的数据的解决方案
  9. 现代语音信号处理之语谱图(spectrogram)
  10. python自动化办公——python操作Excel、Word、PDF集合大全
  11. 动态添加的文本框验证_看腻了普通的PPT封面?手把手教你制作“动态”封面,震惊全场!...
  12. 微信公众号关注自动回复得到小程序链接
  13. 聚合数据iOS SDK 12306火车票查询订票演示示例
  14. 厂级监控系统镜像服务器,厂级监控信息系统(SIS).PDF
  15. Scala——面向对象和函数式编程语言
  16. windows开启远程桌面,防火墙拦截:只允许特定IP远程
  17. 安装python 脚本初稿
  18. 小米2019秋招面试算法题 最少立方数之和 dp与dfs解法
  19. TW使用经验杂谈(四)--UI自动化脚本规范思考
  20. 百度O2O两条腿:百度地图和百度糯米

热门文章

  1. github参与开源项目,fork简介
  2. 虚拟地址、虚拟内存、物理地址、物理内存
  3. 【【51单片机红外遥控小风车】】
  4. java jsch 调用shell_使用Jsch执行Shell脚本
  5. Java使用poi导出数据到excel(包括xls和xlsx两种格式)并通过浏览器下载
  6. 201、基于51单片机智能窗帘设计多功能自动手动定时光控遥控红外防盗窗帘控制系统
  7. python生成斐波那契数列通项公式_Python生成器实现斐波那契数列
  8. 怎么用kali做到ARP攻击
  9. AndroidX和Support库使用记录
  10. sublime3的使用经验与技巧【持续更新】