向现有SQLite表添加约束

gro*_*kky 12 sqlite alter-table check-constraints database-schema entity-framework-core

我正在使用SQLite,它不支持向现有表添加约束.

所以我不能做这样的事情(仅作为一个例子):

ALTER TABLE [Customer]
ADD CONSTRAINT specify_either_phone_or_email
CHECK (([Phone] IS NOT NULL) OR ([Email] IS NOT NULL));
Run Code Online (Sandbox Code Playgroud)

这种情况有没有解决方法?

我知道:

  • 我可以为新表添加约束,但它不是新的(它由我的ORM,EF Core生成)
  • 我可以做一个"表重建"(重命名表,创建新表,复制旧数据,删除临时表)但这看起来真的很复杂

思路

  • 我可以以某种方式将表的副本复制到新表中,并进行一些架构更改吗?
  • 或以某种方式"获取"模式,并在SQL脚本中编辑它,然后添加具有该模式的表?

CL.*_*CL. 11

要制作包含某些架构更改的表的副本,您必须手动创建和复制:

BEGIN;
CREATE TABLE Customer_new (
    [...],
    CHECK ([...])
);
INSERT INTO Customer_new SELECT * FROM Customer;
DROP TABLE Customer;
ALTER TABLE Customer_new RENAME TO Customer;
COMMIT;
Run Code Online (Sandbox Code Playgroud)

要读取模式,请.schema Customersqlite3命令行shell中执行.这为您提供了CREATE TABLE语句,您可以编辑和执行该语句.


要更改表格,您可以使用后门.

首先,阅读实际的表定义(这与你得到的相同.schema):

SELECT sql FROM sqlite_master WHERE type = 'table' AND name = 'Customer';
Run Code Online (Sandbox Code Playgroud)

将CHECK约束添加到该字符串,然后sqlite_master使用PRAGMA writable_schema = 1启用写访问权限; 并将新表定义写入其中:

UPDATE sqlite_master SET sql='...' WHERE type='table' AND name='Customer';
Run Code Online (Sandbox Code Playgroud)

然后重新打开数据库.

警告:这仅适用于不更改表的磁盘格式的更改.如果您确实进行了任何更改记录格式的更改(例如添加/删除字段,或修改rowid,或添加需要内部索引的约束),那么您的数据库将会爆炸.