sqlite外键约束 保证数据一致性

1. 外键约束

在SQLite中,可以通过使用外键(Foreign Key)约束CASCADE 选项来实现通过外键删除相关信息

CASCADE选项是指在主键表中删除记录时,相应的外键表中的相关记录也将被自动删除。

sql 复制代码
-- 创建主键表
CREATE TABLE Persons (
    PersonID INTEGER PRIMARY KEY,
    FirstName TEXT,
    LastName TEXT
);

-- 创建外键表,并指定外键约束
CREATE TABLE Orders (
    OrderID INTEGER PRIMARY KEY,
    OrderNumber TEXT,
    PersonID INTEGER,
    FOREIGN KEY (PersonID) REFERENCES Persons(PersonID) ON DELETE CASCADE
);

-- 插入一些数据
INSERT INTO Persons (PersonID, FirstName, LastName) VALUES (1, 'John', 'Doe');
INSERT INTO Orders (OrderID, OrderNumber, PersonID) VALUES (101, 'A123', 1);
INSERT INTO Orders (OrderID, OrderNumber, PersonID) VALUES (102, 'B456', 1);

-- 查询数据
SELECT * FROM Persons;
SELECT * FROM Orders;

-- 删除Persons表中的记录,由于外键上有CASCADE选项,相关的Orders表中的记录也会被删除
DELETE FROM Persons WHERE PersonID = 1;

-- 再次查询数据
SELECT * FROM Persons;
SELECT * FROM Orders;

在关系型数据库中,外键的约束和操作是定义在表级别的,而不是在记录级别。当使用 ON DELETE CASCADE 时,它指定的是删除主表(Persons 表)中的记录时,如何处理与之关联的外键表(Orders 表)中的记录。

在前面的例子中,当删除 'Persons' 表中 'PersonID' 为 1 的记录时,由于有 ON DELETE CASCADE,相关的 'Orders' 表中 'PersonID' 为 1 的记录也会被删除。这是因为在删除 'Persons' 表中 'PersonID' 为 1 的记录时,外键约束会告诉数据库引擎,也要删除 'Orders' 表中相关的记录,以保持数据的一致性。

然而,如果在 'Orders' 表中删除 'PersonID' 的信息,不会直接影响 'Persons' 表。外键关系是单向的。在这个特定的例子中,只有当删除 'Persons' 表中的记录时,才会影响 'Orders' 表中相关的记录。

因此,如果在 'Orders' 表中删除 'PersonID' 的信息,不会直接导致 'Persons' 表中相关的记录被删除。只有在删除 'Persons' 表中 'PersonID' 的记录时,由于 ON DELETE CASCADE,才会影响 'Orders' 表中相关的记录。

2. 一个表中含有多个外键约束

一个表可以与其他表建立多个关联关系,每个关联关系都需要通过外键来实现。每个外键约束都描述了表与表之间的关系。

每个外键约束的操作是相互独立的,对一个外键的操作不会影响其他外键的操作。

sql 复制代码
CREATE TABLE Persons (
    PersonID INTEGER PRIMARY KEY,
    FirstName TEXT,
    LastName TEXT
);

CREATE TABLE Orders (
    OrderID INTEGER PRIMARY KEY,
    OrderNumber TEXT,
    PersonID INTEGER,
    FOREIGN KEY (PersonID) REFERENCES Persons(PersonID) ON DELETE CASCADE ON UPDATE NO ACTION
);

CREATE TABLE Invoices (
    InvoiceID INTEGER PRIMARY KEY,
    InvoiceNumber TEXT,
    CustomerID INTEGER,
    VendorID INTEGER,
    FOREIGN KEY (CustomerID) REFERENCES Persons(PersonID) ON DELETE SET NULL ON UPDATE CASCADE,
    FOREIGN KEY (VendorID) REFERENCES Persons(PersonID) ON DELETE SET NULL ON UPDATE CASCADE
);

在上述例子中,Invoices 表包含两个外键约束,分别与 Persons 表的两个关联关系相关联。Invoices 表中的 CustomerID 列和 VendorID 列都是对应于 Persons 表中的 PersonID 列的外键。

第一个外键约束将 CustomerID 列与 Persons 表的 PersonID 列关联,并指定了 ON DELETE SET NULL 和 ON UPDATE CASCADE。

第二个外键约束将 VendorID 列与 Persons 表的 PersonID 列关联,并同样指定了 ON DELETE SET NULL 和 ON UPDATE CASCADE。

3. 多表之间的外键约束

在标准的 SQL 外键约束中,删除 A 表的一条记录时,如果 B 表有对 A 表的外键约束,并且设置了 ON DELETE CASCADE,则会自动删除 B 表中与 A 表相关的记录。如果 C 表有对 B 表的外键约束,同样设置了 ON DELETE CASCADE,那么在删除 A 表的记录时,会先触发删除 B 表的记录,然后会自动删除 C 表中与 B 表相关的记录。

sql 复制代码
CREATE TABLE A (
    A_ID INTEGER PRIMARY KEY
);

CREATE TABLE B (
    B_ID INTEGER PRIMARY KEY,
    A_ID INTEGER,
    FOREIGN KEY (A_ID) REFERENCES A(A_ID) ON DELETE CASCADE
);

CREATE TABLE C (
    C_ID INTEGER PRIMARY KEY,
    B_ID INTEGER,
    FOREIGN KEY (B_ID) REFERENCES B(B_ID) ON DELETE CASCADE
);

-- 插入一些数据
INSERT INTO A (A_ID) VALUES (1);
INSERT INTO B (B_ID, A_ID) VALUES (100, 1);
INSERT INTO C (C_ID, B_ID) VALUES (1000, 100);

-- 查询数据
SELECT * FROM A;
SELECT * FROM B;
SELECT * FROM C;

-- 删除 A 表中的记录,观察 B 表和 C 表的变化
DELETE FROM A WHERE A_ID = 1;

-- 查询数据,您会发现 B 表和 C 表中与 A 表相关的记录都被删除了
SELECT * FROM A;
SELECT * FROM B;
SELECT * FROM C;
相关推荐
2501_938780282 分钟前
《轨道交通检测系统中 Qt 与数据库交互的优化方案》
数据库·qt·交互
qqxhb16 分钟前
系统架构设计师备考第61天——嵌入式系统架构模式&操作系统&数据库&中间件
数据库·中间件·系统架构·sqlite·dds·层次化(封闭/开放)·递归模式
SelectDB28 分钟前
Apache Doris 数据导入原理与性能优化 | Deep Dive
运维·数据库·数据分析
悟能不能悟1 小时前
在service方法中已经catch异常,Transactional失效怎么办
java·数据库·sql
月夜奇术师1 小时前
SQL查询性能优化:从30分钟到30秒的蜕变——破解串行查询瓶颈
数据库·性能优化
zimoyin2 小时前
解决导入的数据库中因为 sql_mode 不同 视图无法打开问题
数据库·sql
程序边界2 小时前
MySQL至KingbaseES迁移最佳实践(上篇):迁移准备与实施规划
数据库·mysql
IvorySQL2 小时前
Oracle 19c数据库迁移到IvorySQL 4.6实战
数据库
kanimito2 小时前
开始改变第六天 MySQL(2)
数据库·mysql
晓py3 小时前
理解 MySQL 架构:从连接到存储的全景视图
数据库·mysql·架构