SQL 外键(Foreign Key)详细讲解

1. 什么是外键?​
  • ​定义​​:外键是数据库表中的一列(或一组列),用于​​建立两个表之间的关联关系​​。外键的值必须匹配另一个表的主键(Primary Key)或唯一约束(Unique Constraint)的值。
  • ​作用​​:
    • 确保数据的​​引用完整性​​(Referential Integrity),防止无效数据插入。
    • 维护表之间的逻辑关系(如“一对多”或“多对多”)。

​2. 外键的语法​

在创建表时定义外键:

CREATE TABLE 子表 (列1 数据类型,列2 数据类型,...FOREIGN KEY (外键列) REFERENCES 父表(主键列)[ON DELETE 约束行为] [ON UPDATE 约束行为]
);

在已有表中添加外键:

ALTER TABLE 子表
ADD CONSTRAINT 约束名称
FOREIGN KEY (外键列) REFERENCES 父表(主键列)
[ON DELETE 约束行为] [ON UPDATE 约束行为];

​3. 外键的约束行为​

当父表的记录被删除或更新时,子表的外键如何处理?通过 ON DELETE 和 ON UPDATE 指定:

约束行为说明
​CASCADE​级联操作。父表删除/更新记录时,子表关联记录也被删除/更新。
​SET NULL​父表删除/更新记录时,子表的外键列设为 NULL(要求外键列允许 NULL)。
​NO ACTION​默认行为。阻止父表的删除/更新操作,如果子表存在关联记录。
​RESTRICT​类似 NO ACTION,立即检查约束。
​SET DEFAULT​父表删除/更新记录时,子表的外键设为默认值(需定义默认值)。

​4. 多列外键​

外键可以由多个列组成,需满足:

  • 子表和父表的列数、顺序、数据类型一致。
  • 父表的列必须有唯一约束(如主键或唯一索引)。

​示例​​:

CREATE TABLE 订单详情 (订单ID INT,产品ID INT,数量 INT,PRIMARY KEY (订单ID, 产品ID),FOREIGN KEY (订单ID) REFERENCES 订单(订单ID),FOREIGN KEY (产品ID) REFERENCES 产品(产品ID)
);

​5. 外键的限制与注意事项​
  1. ​父表必须有主键或唯一约束​​。
  2. ​外键列的数据类型必须与父表主键一致​​。
  3. ​引擎支持​​:如 MySQL 的 InnoDB 支持外键,而 MyISAM 不支持。
  4. ​性能影响​​:外键会增加数据操作的检查开销,但能提升数据一致性。
  5. ​循环依赖​​:避免两个表互相引用。

​6. 实际应用示例​

​场景​​:学生表(students)和课程表(courses),通过选课表(enrollments)关联。

-- 父表:学生表
CREATE TABLE students (student_id INT PRIMARY KEY,name VARCHAR(50)
);-- 父表:课程表
CREATE TABLE courses (course_id INT PRIMARY KEY,course_name VARCHAR(50)
);-- 子表:选课表(含外键)
CREATE TABLE enrollments (student_id INT,course_id INT,enrollment_date DATE,FOREIGN KEY (student_id) REFERENCES students(student_id) ON DELETE CASCADE,FOREIGN KEY (course_id) REFERENCES courses(course_id) ON DELETE RESTRICT
);

​插入数据​​:

-- 插入学生和课程
INSERT INTO students VALUES (1, 'Alice');
INSERT INTO courses VALUES (101, 'Math');-- 合法插入:学生和课程存在
INSERT INTO enrollments VALUES (1, 101, '2023-10-01');-- 非法插入:学生不存在,触发外键错误
INSERT INTO enrollments VALUES (999, 101, '2023-10-01'); -- 报错!

​7. 常见问题​
  1. ​外键必须指向主键吗?​
    不,可以指向父表的唯一约束(Unique Constraint)。

  2. ​能否跨数据库引用?​
    通常不支持,外键需在同一数据库内。

  3. ​外键是否允许 NULL?​
    如果外键列允许 NULL,则插入 NULL 是合法的(表示无关联)。

  4. ​如何查看外键约束?​
    使用数据库工具或查询元数据(如 MySQL 的 SHOW CREATE TABLE)。


​8. 总结​
  • ​外键的核心作用​​:维护数据的一致性和关联性。
  • ​适用场景​​:需要强数据完整性的系统(如电商、金融)。
  • ​慎用场景​​:高并发写入且对性能要求极高的系统(需权衡一致性与性能)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/76584.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

5G中的DU和CU的作用

在5G网络架构中,CU(Centralized Unit,集中单元) 和 DU(Distributed Unit,分布单元) 是无线接入网(RAN)的重要组成部分,它们的分工和作用如下: 1.…

深度解析 n8n:强大的开源工作流自动化平台

在数字化时代,企业和个人面临着日益复杂的工作流程和多样化的应用工具,如何高效整合这些资源、实现工作流的自动化成为提升效率的关键。n8n 作为一款开源的工作流自动化平台,凭借其强大的功能、广泛的应用集成能力和灵活的部署方式&#xff0…

ruby超高级语法

以下是 Ruby 中一些 极度硬核 的语法和底层特性,涉及元编程的深渊、虚拟机原理、语法黑魔法等,适用于追求极限的 Ruby 开发者: 高级语法一 一、语法核弹级操作 1. 动态修改继承链 class A; def foo; "A"; end end class B; def …

flutter 获取通话记录和通讯录

Dart SDK version is 3.7.01 dependencies:flutter:sdk: flutterpermission_handler: ^11.0.1 # 权限管理flutter_contacts: ^1.1.92call_log: ^5.0.5cupertino_icons: ^1.0.8dev_dependencies:flutter_test:sdk: flutterflutter_lints: ^5.0.0 2 contact_and_calls_page.da…

bash脚本手动清空mysql表数据

文章目录 1、bash脚本手动清空mysql表数据 1、bash脚本手动清空mysql表数据 #!/bin/bash# 配置区域(修改此处) MYSQL_USER"root" MYSQL_PASSWORD"123456" MYSQL_HOST"localhost" DATABASES("hps-base:base_test_ite…

Spark Core编程

一文读懂Spark Core编程核心要点 最近在学习大数据处理框架Spark,今天来给大家分享一下Spark Core编程中非常重要的内容,包括RDD算子、累加器和广播变量,希望能帮助大家更好地理解和掌握Spark编程。先来说说RDD算子,它是Spark编程…

SDP(一)

SDP(Session Description Protocol)会话描述协议相关参数 Session Description Protocol Version (v): 0 --说明:SDP当前版本号 Owner/Creator, Session Id (o): - 20045 20045 IN IP4 192.168.0.0 --说明:发起者/创建者 会话ID,那么该I…

HarmonyOS:组件布局保存至相册

一,需求背景 有这样一个需求,将页面上的某个自定义组件以图片的形式保存至相册。 二,需求拆解 根据需求分析,可将需求拆解成两步: 1,将组件转换成图片资源; 2,将图片保存到相册…

算法中的数论基础

算法中的数论基础 本篇文章适用于算法考试或比赛之前的临场复习记忆,没有复杂公式推理,基本上是知识点以及函数模版,涵盖取模操作、位运算的小技巧、组合数、概率期望、进制转换、最大公约数、最小公倍数、唯一分解定理、素数、快速幂等知识…

Redis下载稳定版本5.0.4

https://www.redis.net.cn/download/ Redis下载 Redis 版本号采用标准惯例:主版本号.副版本号.补丁级别,一个副版本号就标记为一个标准发行版本,例如 1.2,2.0,2.2,2.4,2.6,2.8,奇数的副版本号用来表示非标准版本,例如2.9.x发行版本是Redis 3.0标准版本的非标准发行版本…

‌UniApp 安卓打包完整步骤(小白向)

‌ ‌一、环境准备‌ ‌安装 HBuilderX‌ 下载最新版 HBuilderX 并安装(官方 IDE,支持一键打包)‌16确保已安装 Node.js(用于依赖管理)‌26 ‌配置 Android 开发环境‌ 安装 ‌Java JDK 17‌(建议选择稳定…

【Springboot知识】Springboot配置加载机制深入解读

文章目录 配置加载概述**Spring Boot 配置加载机制详解****一、配置加载顺序(优先级由低到高)****二、关键配置机制说明****1. Profile 机制****2. 外部化配置****3. 配置属性绑定到 Bean****4. 动态覆盖配置** **三、配置加载流程图****2. 配置导入&…

AI图像生成

要通过代码实现AI图像生成,可以使用深度学习框架如TensorFlow、PyTorch或GANs等技术。下面是一个简单的示例代码,演示如何使用GANs生成手写数字图像: import torch import torchvision import torchvision.transforms as transforms import …

基于springboot的个人博客系统

一、系统架构 前端:html | bootstrap | jquery | css | ajax 后端:springboot | mybatis 环境:jdk1.8 | mysql | maven 二、代码及数据 三、功能介绍 01. 注册 02. 登录 03. 管理后台-首页 04. 管理后台-文章-所有文…

BOTA六维力矩传感器如何打通机器人AI力控操作的三层架构?感知-决策-执行全链路揭秘

想象一下,你对着一个机器人说:“请帮我泡杯茶。”然后,它就真的开始行动了:找茶壶、烧水、取茶叶、泡茶……这一切看似简单,但背后却隐藏着复杂的AI技术。今天,我们就来揭秘BOTA六维力矩传感器在机器人操控…

ffmpeg播放音视频流程

文章目录 🎬 FFmpeg 解码播放流程概览(以音视频文件为例)1️⃣ 创建结构体2️⃣ 打开音视频文件3️⃣ 查找解码器并打开解码器4️⃣ 循环读取数据包(Packet)5️⃣ 解码成帧(Frame)6️⃣ 播放 / …

在 Wireshark 中如何筛选数据包

1. 显示过滤器(Display Filters) 显示过滤器用于 在已捕获的数据包中筛选,语法类似于编程语言中的条件表达式。 (1)基本过滤 表达式说明ip.addr 192.168.1.1显示所有涉及 192.168.1.1 的 IP 包ip.src 192.168.1.1…

ES6 新增特性 箭头函数

简述: ECMAScript 6(简称ES6)是于2015年6月正式发布的JavaScript语言的标准,正式名为ECMAScript 2015(ES2015)。它的目标是使得JavaScript语言可以用来编写复杂的大型应用程序,成为企业级开发语…

Python数据可视化-第7章-绘制3D图表和统计地图

环境 开发工具 VSCode库的版本 numpy1.26.4 matplotlib3.10.1 ipympl0.9.7教材 本书为《Python数据可视化》一书的配套内容,本章为第7章 绘制3D图表和统计地图 本章首先介绍了使用mplot3d工具包绘制3D图表,然后介绍了使用animation模块制作动画&#…

【从零开始学习JVM | 第二篇】HotSpot虚拟机对象探秘

对象的创建 1.类加载检查 虚拟机遇到一条new的指令,首先去检查这个指令的参数能否在常量池中定位到这个类的符号引用,并且检查这个符号引用代表的类是否已被加载过、解析和初始化过。如果没有,那必须先执行类的加载过程。 2.分配内存 在类…