跳到主要内容MySQL 表约束核心指南:从基础到外键实战 | 极客日志SQL
MySQL 表约束核心指南:从基础到外键实战
MySQL 表约束用于保障数据的准确性、唯一性和关联性。本文详解 NULL、DEFAULT、主键、唯一键及外键等核心机制,结合电商订单设计案例,展示如何通过约束减少程序校验逻辑,构建健壮的数据库结构。重点涵盖约束选型避坑指南,帮助开发者设计高效的数据库模型。
在 MySQL 数据库设计中,数据类型决定了字段的存储格式,而表约束则从业务逻辑层面保证数据的合法性和完整性。没有约束的表容易出现空值、重复数据或逻辑冲突(例如学生所属班级不存在),而合理使用约束能让数据库'自我校验',大幅减少程序层的数据验证代码。
表约束核心概念
表约束是对表中字段施加的规则限制,旨在保障数据的准确性、唯一性和关联性。MySQL 支持的核心约束包括:
- 空属性约束(NULL/NOT NULL):控制字段是否允许为空;
- 默认值约束(DEFAULT):未赋值时自动填充预设值;
- 列描述(COMMENT):字段说明,无校验作用;
- 零填充约束(ZEROFILL):数字不足宽度时左侧补 0;
- 主键约束(PRIMARY KEY):唯一标识记录,非空且唯一;
- 自增长约束(AUTO_INCREMENT):整数字段自动递增;
- 唯一键约束(UNIQUE KEY):字段值唯一,允许为空;
- 外键约束(FOREIGN KEY):关联两张表,维护数据一致性。
| 约束类型 | 描述 |
|---|
| 空属性约束(NULL/NOT NULL) | 限制字段是否允许为空值 |
| 默认值约束(DEFAULT) | 字段未赋值时自动使用默认值 |
| 列描述(COMMENT) | 用于字段说明,无校验作用 |
| 零填充约束(ZEROFILL) | 数字类型不足指定宽度时前面填充零 |
| 主键约束(PRIMARY KEY) | 唯一标识每一行记录,非空且唯一 |
| 自增长约束(AUTO_INCREMENT) | 插入新记录时整数字段自动递增 |
| 唯一键约束(UNIQUE KEY) | 保证字段值唯一,通常允许一个空值 |
| 外键约束(FOREIGN KEY) | 关联两张表,保证数据一致性和完整性 |
基础约束:NULL/NOT NULL 与 DEFAULT
基础约束主要控制字段的空值和默认值,是表设计的基石。
空属性约束(NULL/NOT NULL)
NULL 表示字段允许为空(注意:空值无法参与运算,如 1+NULL=NULL);NOT NULL 则强制字段必须赋值。
实战案例: 创建班级表,要求班级名和教室不能为空。
CREATE TABLE myclass (
class_name VARCHAR(20) NOT NULL,
class_room VARCHAR()
);
myclass (, );
myclass(class_name) ();
10
NOT NULL
INSERT INTO
VALUES
'class1'
'301'
INSERT INTO
VALUES
'class2'
默认值约束(DEFAULT)
当字段经常需要固定值时,设置 DEFAULT 可省略该字段插入,系统自动填充。
实战案例: 用户表,年龄默认 0,性别默认'男'。
CREATE TABLE tt10 (
name VARCHAR(20) NOT NULL,
age TINYINT UNSIGNED DEFAULT 0,
sex CHAR(2) DEFAULT '男'
);
INSERT INTO tt10(name) VALUES ('zhangsan');
SELECT * FROM tt10;
注意:NOT NULL 和 DEFAULT 一般不同时显式声明,因为 DEFAULT 已隐含了非空语义。
COMMENT 仅用于描述字段含义,方便开发者和 DBA 理解结构,无实际校验功能。需通过 SHOW CREATE TABLE 查看。
CREATE TABLE tt12 (
name VARCHAR(20) NOT NULL COMMENT '姓名',
age TINYINT UNSIGNED DEFAULT 0 COMMENT '年龄',
sex CHAR(2) DEFAULT '男' COMMENT '性别'
);
DESC tt12;
SHOW CREATE TABLE tt12\G
零填充约束(ZEROFILL)
仅对数字类型有效。当字段值宽度小于设定宽度时,自动在左侧填充 0(仅格式化显示,实际存储值不变)。
CREATE TABLE tt3 (
a INT(5) UNSIGNED ZEROFILL,
b INT(10) UNSIGNED
);
INSERT INTO tt3 VALUES (1, 2);
SELECT * FROM tt3;
SELECT a, HEX(a) FROM tt3;
关键点:无 ZEROFILL 时,数字类型后的宽度(如 INT(10))毫无意义,仅用于显示格式化。
核心约束:主键、自增长与唯一键
这三者是保证数据唯一性的核心,解决'重复数据'和'逻辑主键'问题。
主键约束(PRIMARY KEY)
- 非空(
NOT NULL)且唯一(UNIQUE);
- 一张表最多只能有一个主键;
- 通常用于标识唯一记录(如用户 ID、学号)。
CREATE TABLE tt13 (
id INT UNSIGNED PRIMARY KEY COMMENT '学号(主键)',
name VARCHAR(20) NOT NULL
);
INSERT INTO tt13 VALUES (1, 'aaa');
INSERT INTO tt13 VALUES (1, 'bbb');
当单个字段无法唯一标识记录时,可使用复合主键(多字段联合唯一)。
CREATE TABLE tt14 (
id INT UNSIGNED,
course CHAR(10) COMMENT '课程代码',
score TINYINT UNSIGNED DEFAULT 60 COMMENT '成绩',
PRIMARY KEY(id, course)
);
INSERT INTO tt14(id, course) VALUES (1, '123');
INSERT INTO tt14(id, course) VALUES (1, '123');
ALTER TABLE tt13 ADD PRIMARY KEY(id);
ALTER TABLE tt13 DROP PRIMARY KEY;
自增长约束(AUTO_INCREMENT)
自增长字段会自动从当前最大值 + 1 生成新值。核心特性:
- 必须是整数类型;
- 必须是索引(KEY 一栏有值,通常与主键搭配);
- 一张表最多只能有一个自增长。
CREATE TABLE tt21 (
id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
name VARCHAR(10) NOT NULL DEFAULT ''
);
INSERT INTO tt21(name) VALUES ('a');
INSERT INTO tt21(name) VALUES ('b');
SELECT * FROM tt21;
SELECT LAST_INSERT_ID();
唯一键约束(UNIQUE KEY)
唯一键用于保证字段值唯一,但允许为空(空值不参与唯一性比较),解决'一张表多个唯一字段'的需求(主键仅能有一个)。
| 特性 | 主键(PRIMARY KEY) | 唯一键(UNIQUE KEY) |
|---|
| 唯一性 | 是 | 是 |
| 非空性 | 是 | 否(允许空值) |
| 一张表数量 | 最多 1 个 | 多个 |
| 核心作用 | 标识唯一记录 | 保证业务字段不重复 |
CREATE TABLE student (
id CHAR(10) UNIQUE COMMENT '学号(唯一,可空)',
name VARCHAR(10)
);
INSERT INTO student(id, name) VALUES ('01', 'aaa');
INSERT INTO student(id, name) VALUES (NULL, 'bbb');
INSERT INTO student(id, name) VALUES ('01', 'ccc');
关联约束:外键(FOREIGN KEY)
外键用于定义主表和从表的关联关系,保证数据的逻辑一致性(如学生的班级必须存在于班级表中)。
外键核心规则
- 外键定义在从表上,主表必须有主键或唯一键;
- 从表外键字段的值必须在主表对应字段中存在,或为 NULL;
- 主表删除/修改关联记录时,需处理从表关联数据(如级联删除、拒绝操作)。
实战案例
创建班级表(主表)和学生表(从表),学生表的 class_id 关联班级表的 id。
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)
);
INSERT INTO myclass VALUES (10, 'C++大牛班'), (20, 'Java大神班');
INSERT INTO stu VALUES (100, '张三', 10), (101, '李四', 20);
INSERT INTO stu VALUES (102, '王五', 30);
INSERT INTO stu VALUES (102, '赵六', NULL);
外键的意义
外键的核心价值是'让数据库自动校验数据关联性',避免出现逻辑冲突(如不存在的班级、不存在的用户订单)。若不设置外键,需在程序中手动校验,增加开发成本且易出错。
综合实战:设计电商订单表
结合所有约束,设计电商系统的商品表、客户表、购买表,满足以下需求:
- 商品表:商品编号自增主键,名称非空,单价默认 0;
- 客户表:客户编号自增主键,姓名非空,邮箱唯一,性别枚举(男/女),身份证唯一;
- 购买表:订单号自增主键,关联客户编号和商品编号(外键),购买数量默认 0。
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 FLOAT NOT NULL DEFAULT 0.1 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 COMMENT '电子邮箱(唯一)',
sex ENUM('男','女') NOT NULL COMMENT '性别',
card_id CHAR(18) UNIQUE 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 1 COMMENT '购买数量',
FOREIGN KEY(customer_id) REFERENCES customer(customer_id),
FOREIGN KEY(goods_id) REFERENCES goods(goods_id)
);
约束选型避坑指南和总结
- 优先使用非空约束:尽量让字段
NOT NULL,空值会导致查询条件(如 WHERE age=0)失效,且无法参与运算;
- 主键选择逻辑 ID:主键建议用与业务无关的自增整数(如
id),避免用身份证、手机号等业务字段(需频繁修改);
- 唯一键保证业务唯一性:如邮箱、身份证等业务字段,用
UNIQUE KEY 约束,而非主键;
- 外键谨慎使用:外键会降低表的插入/更新性能,高并发场景可去掉外键,在程序中校验关联性;
- 自增长字段注意重置:删除表中数据后,自增长值不会自动重置,需用
ALTER TABLE 表名 AUTO_INCREMENT=1 手动重置。
MySQL 表约束是数据完整性的核心保障,从基础的空值/默认值约束,到核心的主键/唯一键约束,再到关联的外键约束,各自解决不同场景的问题:
- 基础约束(NULL/DEFAULT/COMMENT/ZEROFILL):控制字段基础属性;
- 核心约束(PRIMARY KEY/AUTO_INCREMENT/UNIQUE KEY):保证数据唯一性;
- 关联约束(FOREIGN KEY):保证表间数据逻辑一致。
合理组合约束能让数据库表结构更健壮,减少程序中的数据校验代码,同时提升数据可靠性。
相关免费在线工具
- SQL 美化和格式化
在线格式化和美化您的 SQL 查询(它支持各种 SQL 方言)。 在线工具,SQL 美化和格式化在线工具,online
- SQL转CSV/JSON/XML
解析 INSERT 等受限 SQL,导出为 CSV、JSON、XML、YAML、HTML 表格(见页内语法说明)。 在线工具,SQL转CSV/JSON/XML在线工具,online
- CSV 工具包
CSV 与 JSON/XML/HTML/TSV/SQL 等互转,单页多 Tab。 在线工具,CSV 工具包在线工具,online
- Base64 字符串编码/解码
将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online
- Base64 文件转换器
将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online
- Markdown转HTML
将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online