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;
相关推荐
weixin_4414552627 分钟前
Mysql MVCC
数据库·mysql
Su-RE1 小时前
【ElasticSearch】text 和 keyword 类型区分
java·数据库·elasticsearch
武子康1 小时前
Java-146 深入浅出 MongoDB 数据插入、批量写入、BSON 格式与逻辑查询and or not操作指南
java·开发语言·数据库·sql·mongodb·性能优化·nosql
奥尔特星云大使1 小时前
MySQL快速构建主从(基于GTID)
数据库·mysql·主从复制
小园子的小菜1 小时前
MySQL ORDER BY 深度解析:索引排序规则与关键配置参数阈值
数据库·mysql
wxjlkh1 小时前
Oracle Exadata一体机简介 1千多个W
数据库·oracle
泽虞1 小时前
《Qt应用开发》笔记p3
linux·开发语言·数据库·c++·笔记·qt·面试
XXYBMOOO1 小时前
如何自定义 Qt 日志处理并记录日志到文件
开发语言·数据库·qt
不剪发的Tony老师2 小时前
PEV2:一款PostgreSQL执行计划可视化工具
数据库·postgresql
IT 小阿姨(数据库)2 小时前
PostgreSQL wal_e 工具详解
运维·数据库·sql·postgresql·centos