3人参与 • 2026-03-20 • Mysql
在前几篇文章中,我们系统学习了 sql 的 ddl、dml 和 dql 核心用法,掌握了数据库、表、数据的增删改查。但在实际开发中,仅靠数据类型无法保证数据的完整性和有效性 —— 比如学号不能为空且不能重复、姓名必须填写、手机号不能重复等,这就需要约束(constraint) 来实现。
本文将聚焦 sql 中的单表约束,详细讲解主键、非空、唯一、默认约束的语法格式与实战用法,结合案例帮你彻底掌握,让数据更规范!
约束是在数据类型基础上,对数据表中列的值进行进一步的限定,目的是保证数据的完整性、唯一性、有效性,避免脏数据产生。
表格
| 约束类型 | 作用域 | 具体约束 | 核心特点 |
|---|---|---|---|
| 单表约束 | 单张表内 | primary key(主键约束) | 非空 + 唯一,可结合 auto_increment 自增 |
| not null(非空约束) | 列值不能为 null | ||
| unique(唯一约束) | 列值必须唯一,可 null | ||
| default(默认约束) | 未赋值时使用默认值 | ||
| 多表约束 | 多张表间 | foreign key(外键约束) | 关联多表,保证数据一致性(后续单独讲解) |
-- 重置演示数据库(避免干扰) drop database if exists day02; create database day02; use day02;
📌 执行效果:删除已存在的 day02 数据库,重新创建并切换到该数据库。
auto_increment(自增)使用,简化数据插入。表格
| 应用场景 | 语法格式 | 说明 |
|---|---|---|
| 建表时列级定义 | 列名 数据类型 primary key | 直接在列后添加,简洁常用 |
| 建表时表级定义 | primary key (列名 1, 列名 2...) | 支持复合主键(多列联合主键) |
| 建表后添加 | alter table 表名 add primary key (列名); | 适用于已有表补充主键 |
| 删除主键 | alter table 表名 drop primary key; | 单表仅能有一个主键,删除无需指定列名 |
| 添加自增 | alter table 表名 modify 列名 数据类型 auto_increment; | 自增仅支持整型列,且需绑定主键 |
场景 1:建表时添加主键约束
-- 创建学生表(sid:学号,主键;name:姓名;age:年龄)
drop table if exists student;
create table student(
sid int primary key, -- 方式1:列级定义主键(推荐)
name varchar(10),
age int
-- primary key(sid) -- 方式2:表级定义主键(效果相同)
);📌 执行效果:删除已存在的 student 表,创建新表并将 sid 列设为主键。
-- 查看表结构(可看到sid的key列显示pri,代表主键) desc student;
📌 执行效果:展示 student 表的字段结构,sid 字段的 key 列显示「pri」,标识为主键。
-- 插入测试数据 insert into student values(1, '张三', 18); -- 正常插入
📌 执行效果:成功插入 1 条数据,sid=1、name = 张三、age=18。
insert into student values(1, '张三', 18); -- 报错:主键重复(duplicate entry)
📌 执行效果:执行报错,提示「duplicate entry '1' for key'student.primary'」,主键值不能重复。
insert into student values(null, '张三', 18); -- 报错:主键非空(column 'sid' cannot be null)
📌 执行效果:执行报错,提示「column'sid' cannot be null」,主键列不能为 null。
-- 查看数据 select * from student;
📌 执行效果:仅展示 sid=1 的那条学生数据。
场景 2:建表后添加主键 + 自增
-- 先删除已有的主键约束 alter table student drop primary key;
📌 执行效果:成功删除 student 表的主键约束。
-- 重新添加主键约束,并增加自增功能 alter table student add primary key(sid); alter table student modify sid int auto_increment;
📌 执行效果:为 sid 列重新添加主键约束,并开启自增特性。
-- 再次插入数据(自增特性生效) insert into student values(2, '李四', 20); -- 正常插入
📌 执行效果:成功插入 sid=2 的李四数据。
insert into student values(2, '李四', 20); -- 报错:主键重复
📌 执行效果:执行报错,主键值 2 已存在,无法重复插入。
insert into student values(null, '李四', 20); -- 正常插入(sid自动递增为3)
📌 执行效果:成功插入数据,sid 自动递增为 3。
insert into student values(10, '王五', 20); -- 插入sid=10
📌 执行效果:成功插入 sid=10 的王五数据。
insert into student values(null, '赵六', 66); -- sid自动递增为11
📌 执行效果:成功插入数据,sid 自动递增为 11(基于上一个最大值 10)。
-- 查看最终数据 select * from student;
📌 执行效果:展示 sid 为 1、2、3、10、11 的 5 条学生数据。
场景 3:实际开发推荐写法(建表时直接加主键 + 自增)
-- 重建学生表(推荐写法)
drop table if exists student;
create table student(
sid int primary key auto_increment, -- 主键+自增一步到位
name varchar(10),
age int
);📌 执行效果:删除旧表,创建新表,sid 列同时设置为主键并开启自增。
-- 插入数据(主键列传null即可自动递增) insert into student values(null, '乔峰', 38); insert into student values(null, '虚竹', 30);
📌 执行效果:成功插入 2 条数据,sid 自动递增为 1、2。
-- 查看数据 select * from student;
📌 执行效果:展示 sid=1(乔峰)、sid=2(虚竹)的两条数据。
关键补充:delete vs truncate 对主键自增的影响
表格
| 操作 | 语法格式 | 对主键自增的影响 | 本质 |
|---|---|---|---|
| delete | delete from 表名; | 不重置自增 id | 仅删除数据行 |
| truncate | truncate table 表名; | 重置自增 id 为 1 | 删表后重建同名表 |
-- delete:删除数据,不重置主键自增id delete from student; insert into student values(null, '段誉', 25); -- sid会从3开始(而非1)
📌 执行效果:删除所有数据后插入新数据,sid 从 3 开始(原自增最大值为 2)。
-- truncate:清空表并重置主键自增id(相当于删表重建) truncate table student; insert into student values(null, '阿朱', 20); -- sid从1开始
📌 执行效果:清空表并重置自增 id,插入新数据时 sid 从 1 开始。
表格
| 约束类型 | 语法格式 | 核心规则 |
|---|---|---|
| not null | 建表时:列名 数据类型 not null 建表后添加:alter table 表名 modify 列名 类型 not null 建表后删除:alter table 表名 modify 列名 类型 null | 列值必须填写,不能为 null |
| unique | 建表时:列名 数据类型 unique 建表后添加:alter table 表名 add unique (列名) 建表后删除:alter table 表名 drop index 列名 | 列值必须唯一,可 null(多个 null 不冲突) |
| default | 建表时:列名 数据类型 default ' 默认值' 建表后添加:alter table 表名 modify 列名 类型 default ' 默认值' | 插入数据时未赋值,则使用默认值 |
-- 创建员工表(整合非空、唯一、默认约束)
create table employee(
eid int primary key auto_increment, -- 员工id:主键+自增
name varchar(10) not null, -- 姓名:非空约束
mobile varchar(11) unique, -- 手机号:唯一约束
address varchar(50) default '北京' -- 住址:默认约束(默认值为北京)
);📌 执行效果:创建 employee 表,为不同字段分别设置主键 + 自增、非空、唯一、默认约束。
-- 查看表结构(可看到对应约束标识) desc employee;
📌 执行效果:展示 employee 表结构,name 列 null 显示「no」(非空),mobile 列 key 显示「uni」(唯一),address 列 default 显示「北京」。
-- 插入测试数据 -- 案例1:正常插入(所有字段赋值) insert into employee values(null, '乔峰', '13800138000', '南院大王府');
📌 执行效果:成功插入 1 条员工数据,eid 自动递增为 1。
-- 案例2:姓名为空(报错:非空约束) -- insert into employee values(null, null, '13800138001', '缥缈峰');
📌 执行效果:执行报错,提示「column 'name' cannot be null」,非空约束生效。
-- 案例3:手机号重复(报错:唯一约束) -- insert into employee values(null, '乔峰', '13800138000', '缥缈峰');
📌 执行效果:执行报错,提示「duplicate entry '13800138000' for key 'employee.mobile'」,唯一约束生效。
-- 案例4:手机号不重复,正常插入 insert into employee values(null, '阿紫', '13800138001', '星宿海');
📌 执行效果:成功插入 1 条数据,eid 自动递增为 2。
-- 案例5:未指定列名,值个数不足(报错) -- insert into employee values(null, '段誉', '13800138002');
📌 执行效果:执行报错,提示「column count doesn't match value count」,字段数量不匹配。
-- 案例6:指定列插入,住址使用默认值 insert into employee(eid, name, mobile) values(null, '段誉', '13800138002');
📌 执行效果:成功插入数据,address 列自动填充默认值「北京」,eid 递增为 3。
-- 查看最终数据(段誉的address会显示默认值“北京”) select * from employee;
📌 执行效果:展示 3 条员工数据,段誉的 address 字段值为「北京」(默认值)。
表格
| 操作 | 语法示例 |
|---|---|
| 删除主键 | alter table student drop primary key; |
| 添加非空约束 | alter table employee modify name varchar(10) not null; |
| 删除非空约束 | alter table employee modify name varchar(10) null; |
| 添加唯一约束 | alter table employee add unique(mobile); |
| 删除唯一约束 | alter table employee drop index mobile; |
| 添加默认约束 | alter table employee modify address varchar (50) default ' 上海 '; |
auto_increment 可自动递增,建表时「列名 类型 primary key」是最常用写法;not null 强制列值非空,unique 强制列值唯一,default 为列设置默认值,三者均支持建表时定义、建表后修改;本文掌握了单表约束的语法格式与核心用法,后续将讲解多表约束(外键约束) 和多表查询,这是 sql 进阶的关键,关注我,持续解锁 sql 实战技能!
到此这篇关于mysql单表约束超详细实战用法(主键 / 非空 / 唯一 / 默认)的文章就介绍到这了,更多相关mysql单表约束内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
您想发表意见!!点此发布评论
版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。
发表评论