#1、约束的概述
#1.1为什么约束
#数据完整性是指数据的精确性和可靠性他可以防止数据库中存在不符合语义规定和防止因错误信息的输入输出或无效操作或错误信息而提出的实体完整性(Entity Integrity) :例如,同一个表中,不能存在两条完全相同无法区分的记录#域完整性(Domain Integrity) :例如:年龄范围0-120,性别范围“男/女”
#引用完整性(Referential Integrity) :例如:员工所在部门,在部门表中要能找到这个部门
#用户自定义完整性(User-defined Integrity) :例如:用户名唯一、密码不能为空等,本部门经理的工资不得高于本部门职工的平均工资的5倍。
#1.2 什么叫约束?对表中字段的限制
#1.3约束的分类
#角度1:约束字段的个数#单列约束 vs 多列约束
#角度2:约束的作用范围#列级约束:将此约束声明在对应字段的后面#表级约束:在表中所以字段都声明完,在所以字段的后面的约束#角度3:约束的作用(或功能)
① not null (非空约束)
② unique (唯一性约束)
③ primary (主键约束)
④ FOREIGN KEY 外键约束
⑤ check (检查约束)
⑥ default (默认值约束)
1.4 如何添加约束?
CREATE TABLE 时添加约束ALTER TABLE 时增加约束、删除约束
#2.如何查看表中的约束
SELECT * FROM information_schema.table_constraints
WHERE TABLE_NAME = 'employees';
#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));
ALTER TABLE test1
MODIFY last_name VARCHAR(15) NOT NULL;
DESC test1;
INSERT INTO test1VALUES(NULL,'Tom','[email protected]',6666.33);
SELECT * FROM test1;
INSERT INTO test1VALUES(2,NULL,'[email protected]',6999.33);
#搜索默认值,因为没有默认值,则last_name会为空,但是不能为空,就会报错
INSERT INTO test1(id,email)VALUES(1,'[email protected]');
#3.2在ALTER TABLE 时添加约束
#如果表中已经含有 NULL 值,则会添加不成功
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));
#在创建唯一约束的时候,如果不给唯一约束命名,就默认和列名相同
SELECT * FROM information_schema.table_constraints
WHERE TABLE_NAME = 'test2';
INSERT INTO test2VALUES(1001,'Tom','[email protected]',456789.4);#错误
INSERT INTO test2VALUES(1001,'Jom','[email protected]',456789.4);
#可以向声明为UNIQUE的字段上添加NULL值
#并且可以多次添加 NULL 值
INSERT INTO test2
VALUES(1002,'Jom',NULL,456789.4);
#4.2 在ALTER TABLE 时添加约束
ALTER TABLE test2
ADD CONSTRAINT uk_test2_sal UNIQUE(salary);
ALTER TABLE test2
MODIFY last_name VARCHAR(15) UNIQUE;
#4.3 复合的唯一性约束
#只要不是全部相同,就可以添加成功
CREATE TABLE USER(id INT,`name` VARCHAR(15),`password` VARCHAR(15),
#表级约束
CONSTRAINT uk_user_name_pwd UNIQUE(`name`,`password`));
INSERT INTO USERVALUES(1,'Tom','abc');
INSERT INTO USERVALUES(1,'Tom','abcd');
SELECT * FROM USER;
DESC USER;
#4.4删除唯一性约束
#添加唯一性约束的列上也会自动创建唯一索引。
#删除唯一约束只能通过删除唯一索引的方式删除。
#删除时需要指定唯一索引名,唯一索引名就和唯一约束名一样。
#如果创建唯一约束时未指定名称,如果是单列,就默认和列名相同;如果是组合列,那么默认和()
#中排在第一个的列名相同。也可以自定义唯一性约束名。
#4.4 如何删除唯一性索引
ALTER TABLE test2
DROP INDEX last_name;
ALTER TABLE test2
DROP INDEX uk_test2_sal;
DESC test2;
#5 primary key 主键约束 非空且唯一
#5.1 在CREATE TABLE 添加约束
#一个表里只能添加一个主键约束
CREATE TABLE test3(id INT PRIMARY KEY ,
#列级约束
last_name VARCHAR(15),salary DECIMAL(10,2),email VARCHAR(25)
#表及约束
CONSTRAINT pk_test5_id PRIMARY KEY (id));
#MySQL的主键名总是PRIMARY,就算自己命名了主键约束名也没有
CREATE TABLE user1(id INT,NAME VARCHAR(25),PASSWORD VARCHAR(25),
#表级约束
PRIMARY KEY(id,NAME));
#只要两个都不相同,就可以添加成功
INSERT INTO user1
VALUES(1,'Tom','123456');
INSERT INTO user1
VALUES(2,'Tom','123456');
INSERT INTO user1
VALUES(1,'Jom','123456');
#两个其一不能为 null
INSERT INTO user1
VALUES(NULL,‘Tom’,‘123456’);
#5.2 在 ALTER TABLE 时添加约束
CREATE TABLE test6(id INT,last_name VARCHAR(15),salary DECIMAL(10,2),email VARCHAR(25));
ALTER TABLE test6
ADD PRIMARY KEY(id);
ALTER TABLE test6
MODIFY id INT PRIMARY KEY;
#5.3 删除主键约束
#不需要指定主键名,因为一个表只有一个主键,删除主键后,非空仍然存在
ALTER TABLE test6
DROP PRIMARY KEY;
#6.自增列,AUTO_INCREMENT
#一个表里只能有一个自增列#自增列必须是 主键 或者 唯一 约束 整数类型开发中,一旦主键作用的字段上声明有自增列,则我们添加数据时就不要给主键添加数据了
CREATE TABLE test7
(id INT PRIMARY KEY AUTO_INCREMENT,last_name VARCHAR(25));
#但当我们向主键(含自增)的字段上添加0或null时实际上回自动的往上添加指定数值
INSERT INTO test7(last_name)
VALUES('Tom');
#当我们添加一个自增列没有的数值,会添加成功#添加的数值大于自增列最大的,在继续添加时,会在最大的值后面添加
INSERT INTO test7(id,last_name)
VALUES(10,'Tom');
#6.2在ALTER TABLE 时添加自增列
CREATE TABLE test8
(id INT PRIMARY KEY ,last_name VARCHAR(25));
ALTER TABLE test8
MODIFY id INT AUTO_INCREMENT;
#6.2在ALTER TABLE 时删除自增列
ALTER TABLE test8MODIFY id INT ;
#6.3 MySQL 8.0新特性—自增变量的持久化
#7 foreign key (外键约束)
#7.1 在CREATE TABLE 时添加约束#创建主表、从表
①创建主表CREATE TABLE dept1
(dept_id INT ,dept_name VARCHAR(25));
②创建从表CREATE TABLE emp1
(emp_id INT PRIMARY KEY AUTO_INCREMENT,emp_name VARCHAR(25),department_id INT,
#表级约束
CONSTRAINT fk_emp1_dept1_id FOREIGN KEY (department_id)
REFERENCES dept1(dept_id));
#上述操作报错,因为主表中的dept1上没有主键约束或唯一性约束
#添加
ALTER TABLE dept1
MODIFY dept_id INT PRIMARY KEY;
#7.2演示外键的效果
#错误,因为主表没有 10 部门
INSERT INTO emp1VALUES (1001,'Tom',10);
#在主表中添加 10 部门
INSERT INTO dept1VALUES(10,'IT');
INSERT INTO emp1VALUES (1001,'Tom',10);
#不能删除 10 部门 ,因为从表有约束,需要先删除从表含 10部门的数据
DELETE TABLE dept1
WHERE dept_id = 10;
#7.1 在ALTER TABLE 时添加外键约束
CREATE TABLE dept2(dept_id INT PRIMARY KEY,dept_name VARCHAR(25));
CREATE TABLE emp2(emp_id INT PRIMARY KEY AUTO_INCREMENT,emp_name VARCHAR(25),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.2约束等级
Cascade方式 :在父表上update/delete记录时,同步update/delete掉子表的匹配记录Set null方式 :在父表上update/delete记录时,将子表上匹配记录的列设为null,但是要注意子表的外键列不能为not null No action方式 :如果子表中有匹配的记录,则不允许对父表对应候选键进行update/delete操作Restrict方式 :同no action, 都是立即检查外键约束Set default方式 (在可视化工具SQLyog中可能显示空白):父表有变更时,子表将外键列设置成一个默认的值,但Innodb不能识别如果没有指定等级,就相当于Restrict方式
#7.3删除外键约束#一个表中可以声明有多个外键约束
ALTER TABLE emp1
DROP FOREIGN KEY fk_emp1_dept1_id;
SELECT * FROM information_schema.table_constraints
WHERE TABLE_NAME = 'emp1';
#在手动的删除外键约束对应的普通索引
SHOW INDEX FROM emp1;
ALTER TABLE emp1
DROP INDEX fk_emp1_dept1_id;
#8.CHECK约束#MySQL5.7不支持CHECK约束,MySQL8.0支持CHECK约束
CREATE TABLE test
(id INT,last_name VARCHAR(25),salary DECIMAL(10,2) CHECK(salary > 2000));
INSERT INTO testVALUES(1,'Tom',2500);
#添加失败
INSERT INTO testVALUES(2,'Tom',1500);
#9 DEFAULT 约束#9.1 在 CREATE TABLE 中添加约束
CREATE TABLE test11
(id INT,last_name VARCHAR(25),salary DECIMAL(10,2) DEFAULT 2000);
INSERT INTO test11
VALUES(1,'Tom',5000);
#salary 为 null
INSERT INTO test11VALUES(2,'Tom',NULL);
#salary 没有进行输入,则是默认值,为2000
INSERT INTO test11(id,last_name)
VALUES(3,'Mike');
#9.2 在 ALTER TABLE 中添加约束
CREATE TABLE test12
(id INT,last_name VARCHAR(25),salary DECIMAL(10,2));
ALTER TABLE test12
MODIFY salary DECIMAL(10,2) DEFAULT 2000;
#9.3 在 ALTER TABLE 中删除约束
ALTER TABLE test12
MODIFY salary DECIMAL(10,2);
SELECT * FROM test;
SELECT * FROM emp1;
DESC test12;
练习
#练习1
CREATE DATABASE test04_emp;
USE test04_emp;
CREATE TABLE emp2
(id INT,emp_name VARCHAR(15));
CREATE TABLE dept2
(id INT,dept_name VARCHAR(15));
DESC emp2;
DESC dept2;
#1.向表emp2的id列中添加PRIMARY KEY约束
ALTER TABLE emp2
#add primary key(id);
MODIFY id INT PRIMARY KEY;
#删除主键约束
ALTER TABLE emp2
DROP PRIMARY KEY;
#2. 向表dept2的id列中添加PRIMARY KEY约束
ALTER TABLE dept2
MODIFY id INT PRIMARY KEY;
#3. 向表emp2中添加列dept_id,并在其中定义FOREIGN KEY约束,与之相关联的列是dept2表中的id列
ALTER TABLE emp2
ADD dept_id INT ;
ALTER TABLE emp2
ADD CONSTRAINT fk_emp2_dept2_id FOREIGN KEY(dept_id) REFERENCES dept2(id);
#练习2
#1. 创建数据库test04_company
CREATE DATABASE test04_company;
USE test04_company;
#2. 按照下表给出的表结构在test04_company数据库中创建两个数据表offices和employees
CREATE TABLE offices(
officeCode INT PRIMARY KEY,
city VARCHAR(50) NOT NULL,
address VARCHAR(50),
country VARCHAR(50) NOT NULL,
postalCode VARCHAR(15) UNIQUE);
CREATE TABLE employees(
employeeNumber INT(11) PRIMARY KEY AUTO_INCREMENT,
lastName VARCHAR(50) NOT NULL,
firstName VARCHAR(50) NOT NULL,
mobile VARCHAR(25) UNIQUE,
officeCode INT NOT NULL,
jobTitle VARCHAR(50) NOT NULL,birth DATETIME UNIQUE,
note VARCHAR(255),
sex VARCHAR(5),
CONSTRAINT fk_emp_offcode FOREIGN KEY (officeCode) REFERENCES offices(officeCode));
#3. 将表employees的mobile字段修改到officeCode字段后面
ALTER TABLE employees
MODIFY mobile VARCHAR(25) AFTER officeCode;
#4. 将表employees的birth字段改名为employee_birth
ALTER TABLE employees
CHANGE birth employee DATETIME;
#5. 修改sex字段,数据类型为CHAR(1),非空约束
ALTER TABLE employees
MODIFY sex CHAR(1) NOT NULL;
#6. 删除字段note
ALTER TABLE employees
DROP COLUMN note;
#7. 增加字段名favoriate_activity,数据类型为VARCHAR(100)
ALTER TABLE employees
ADD favoriate_activity VARCHAR(100) ;
#8. 将表employees名称修改为employees_info
RENAME TABLE employees
TO employees_info;