外键基础概念
外键是一个表中的字段,它引用另一个表的主键或唯一键,用于确保两个表之间的数据一致性。
sql
-- 主表:用户
CREATE TABLE user (
id INT PRIMARY KEY AUTO_INCREMENT,
name VARCHAR(50) NOT NULL
);
-- 从表:订单(通过 user_id 关联 user.id)
CREATE TABLE orders (
id INT PRIMARY KEY AUTO_INCREMENT,
order_no VARCHAR(20) NOT NULL,
user_id INT,
FOREIGN KEY (user_id) REFERENCES user(id) -- 外键约束
);
本质是数据库层面的强制引用完整性检查。
外键约束类型与行为
当主表数据变更时,从表如何响应?MySQL 提供 5 种约束行为:
| 约束类型 | 主表 DELETE | 主表 UPDATE | 使用场景 |
|---|---|---|---|
| CASCADE | 级联删除从表数据 | 级联更新从表外键值 | 强关联数据(用户注销→删除订单) |
| SET NULL | 从表外键设为 NULL | 从表外键设为 NULL | 可选关联(部门删除→员工部门ID置空) |
| RESTRICT | 阻止删除(默认行为) | 阻止更新 | 强保护(有订单时不能删用户) |
| NO ACTION | 同 RESTRICT(SQL 标准) | 同 RESTRICT | 与 RESTRICT 等价 |
| SET DEFAULT | 设为默认值(InnoDB 不支持) | 设为默认值 | InnoDB 引擎无效 |
完整创建语法:
sql
CREATE TABLE orders (
id INT PRIMARY KEY,
user_id INT,
FOREIGN KEY (user_id)
REFERENCES user(id)
ON DELETE CASCADE -- 删除用户时,级联删除订单
ON UPDATE RESTRICT -- 更新用户ID时,阻止(一般不会更新主键)
);
外键实现原理
数据字典与元数据
sql
-- 查看外键信息
SELECT
CONSTRAINT_NAME,
TABLE_NAME,
COLUMN_NAME,
REFERENCED_TABLE_NAME,
REFERENCED_COLUMN_NAME
FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE
WHERE REFERENCED_TABLE_NAME IS NOT NULL;
InnoDB 在数据字典中维护外键元数据,包括父表、子表、关联列和约束类型。
锁机制(性能杀手)
当插入/更新/删除子表时,InnoDB 会加间隙锁(Gap Lock)保护父表:
sql
-- 场景:orders 表有外键 user_id 引用 user.id
BEGIN;
INSERT INTO orders (user_id) VALUES (100); -- 子表插入
-- InnoDB 隐式加锁:
-- 在 user 表的 id=100 记录上加 **共享间隙锁**
-- 阻止其他事务删除/更新 user.id=100,防止插入孤儿订单
锁范围:
- 主表:user (id 列)
- 插入子表:给主表对应 id 加 共享锁
- 删除主表:检查从表是否有引用,加 共享锁 扫描
- 更新主表 id:给新/旧 id 都加锁
性能影响:
- 高并发写入时,外键检查导致锁竞争加剧
- 删除主表记录需全表扫描从表(无索引时)
外键检查时机
约束检查模式:
ini
-- 会话级设置
SET foreign_key_checks = 0; -- 临时关闭外键检查(危险!)
SET foreign_key_checks = 1; -- 开启(默认)
批量导入数据时:
sql
-- 先关闭外键检查,加速导入
SET foreign_key_checks = 0;
LOAD DATA INFILE 'orders.csv' INTO TABLE orders;
SET foreign_key_checks = 1; -- 完成后务必开启
外键的优点
数据一致性保障
sql
-- 无法插入孤儿记录
INSERT INTO orders (user_id) VALUES (999); -- ERROR 1452: Cannot add or update a child row
-- 无法删除被引用的父记录
DELETE FROM user WHERE id = 100; -- ERROR 1451: Cannot delete or update a parent row
级联操作简化业务代码
sql
-- 删除用户时,自动级联删除订单、地址、积分
ON DELETE CASCADE;
-- 无需手动写多个 DELETE 语句
文档化表关系
外键约束显性定义了表间关系,新开发人员可快速理解数据模型。
外键的缺点(互联网大厂弃用原因)
性能瓶颈
- 锁竞争:高并发下,外键检查间隙锁导致锁等待
- 延迟增加:外键检查增加 5-10% SQL 执行时间
- 级联删除慢:删除主表 1 条记录,可能触发从表百万级级联删除
扩展性差
- 分库分表失效:跨库外键无法检查,ShardingSphere 等中间件不支持外键
- 微服务架构冲突:服务拆分后,跨服务外键无法约束
业务耦合
- 删除主表受阻:业务想软删除用户,但外键阻止
- 热更新困难:修改表结构需先删外键,增加运维复杂度
级联陷阱
sql
-- 误操作:删除 1 个用户,级联删除 10 万订单(无法回滚)
DELETE FROM user WHERE id = 1; -- 瞬间删 10 万数据!
-- 解决:应用层软删除 + 定时任务清理
UPDATE user SET deleted = 1 WHERE id = 1; -- 业务层控制
大厂最佳实践:去外键化
应用层保证一致性
java
@Service
public class UserService {
@Transactional
public void deleteUser(Long userId) {
// 1. 先检查是否有订单
if (orderRepository.countByUserId(userId) > 0) {
throw new BusinessException("用户有订单,不能删除");
}
// 2. 再删除用户
userRepository.deleteById(userId);
}
}
软删除 + 定期归档
sql
-- 用户表增加 deleted 字段
ALTER TABLE user ADD COLUMN deleted TINYINT DEFAULT 0;
-- 删除操作变为更新
UPDATE user SET deleted = 1 WHERE id = 100;
-- 从表订单保留 user_id,业务查询自动过滤
SELECT * FROM orders WHERE user_id IN (SELECT id FROM user WHERE deleted = 0);
分布式事务(跨服务)
java
// 使用 Seata / Saga 模式保证最终一致性
@GlobalTransactional
public void createOrder(Long userId) {
// 1. 调用用户服务(校验用户存在)
userService.checkUserExists(userId);
// 2. 创建订单
orderRepository.save(new Order(userId));
// 3. 异常时全局回滚
}
什么时候该用外键?
推荐使用场景
-
内部管理系统:并发低、数据一致性要求高
sql-- 企业内部 ERP CREATE TABLE employee_dept ( emp_id INT, dept_id INT, FOREIGN KEY (emp_id) REFERENCES employee(id) ON DELETE CASCADE ); -
强关联数据:无孤立记录可能
sql-- 订单明细与订单头 FOREIGN KEY (order_id) REFERENCES order_header(id) ON DELETE CASCADE -
开发规范严格:团队小、运维规范
不推荐场景
- 高并发互联网业务:C2C 电商、社交
- 分库分表架构:ShardingSphere、MyCat 环境
- 微服务架构:跨服务数据无法约束
- 需要柔性事务:允许短暂不一致
性能对比测试
sql
-- 测试环境:MySQL 8.0, 100 万订单, 10 万用户
-- 场景 1:插入订单(有外键)
INSERT INTO orders (user_id, amount) VALUES (1001, 99.00);
-- 耗时:1.2ms(检查 user.id=1001 是否存在)
-- 场景 2:插入订单(无外键)
INSERT INTO orders (user_id, amount) VALUES (1001, 99.00);
-- 耗时:0.3ms
-- 场景 3:删除用户(有外键 + 级联)
DELETE FROM user WHERE id = 1001;
-- 耗时:500ms(级联删除 10 条订单)
-- 场景 4:删除用户(无外键 + 软删)
UPDATE user SET deleted = 1 WHERE id = 1001;
-- 耗时:0.1ms
性能结论:外键使写入性能降低 30-50%,删除性能降低 10-100 倍。
外键与索引的关系
外键自动创建索引
sql
-- 建外键时,若 user_id 无索引,InnoDB 自动创建
FOREIGN KEY (user_id) REFERENCES user(id);
-- InnoDB 自动:CREATE INDEX user_id ON orders(user_id);
手动删除外键索引
sql
-- ❌ 错误:无法删除被外键使用的索引
DROP INDEX user_id ON orders; -- ERROR
-- ✅ 正确:先删外键,再删索引
ALTER TABLE orders DROP FOREIGN KEY orders_ibfk_1;
DROP INDEX user_id ON orders;