向现有 SQLite 添加约束 table
Add constraint to existing SQLite table
我正在使用 SQLite,doesn't support adding a constraint 到现有的 table。
所以我不能做这样的事情(仅举个例子):
ALTER TABLE [Customer]
ADD CONSTRAINT specify_either_phone_or_email
CHECK (([Phone] IS NOT NULL) OR ([Email] IS NOT NULL));
这种情况有任何解决方法吗?
我知道:
- 我可以为新的 table 添加约束,但它不是新的(它是由我的 ORM、EF Core 生成的)
- 我可以做一个 "table rebuild"(重命名 table、创建新的、复制旧数据、删除临时文件 table)但这看起来真的很复杂
想法
- 我能否以某种方式将 table 的副本复制到新的 table 中,并进行一些架构更改?
- 或 "get" 模式,并在 SQL 脚本中编辑它,然后添加一个 table 与该模式?
要复制带有一些架构更改的 table,您必须手动创建和复制:
BEGIN;
CREATE TABLE Customer_new (
[...],
CHECK ([...])
);
INSERT INTO Customer_new SELECT * FROM Customer;
DROP TABLE Customer;
ALTER TABLE Customer_new RENAME TO Customer;
COMMIT;
要读取架构,请在 sqlite3
command-line shell 中执行 .schema Customer
。
这为您提供了 CREATE TABLE 语句,您可以编辑和执行它。
要更改table,您可以使用后门。
首先,阅读实际的 table 定义(这与您从 .schema
中得到的相同):
SELECT sql FROM sqlite_master WHERE type = 'table' AND name = 'Customer';
将您的 CHECK 约束添加到该字符串,然后使用 PRAGMA writable_schema=1; 启用对 sqlite_master
的写入访问权限,并将新的 table 定义写入其中:
UPDATE sqlite_master SET sql='...' WHERE type='table' AND name='Customer';
然后重新打开数据库。
警告:这仅适用于不改变 table 的磁盘格式的更改。如果您确实进行了任何更改记录格式的更改(例如 adding/removing 字段,或修改 rowid,或添加需要内部索引的约束),您的数据库将会崩溃。
我正在使用 SQLite,doesn't support adding a constraint 到现有的 table。
所以我不能做这样的事情(仅举个例子):
ALTER TABLE [Customer]
ADD CONSTRAINT specify_either_phone_or_email
CHECK (([Phone] IS NOT NULL) OR ([Email] IS NOT NULL));
这种情况有任何解决方法吗?
我知道:
- 我可以为新的 table 添加约束,但它不是新的(它是由我的 ORM、EF Core 生成的)
- 我可以做一个 "table rebuild"(重命名 table、创建新的、复制旧数据、删除临时文件 table)但这看起来真的很复杂
想法
- 我能否以某种方式将 table 的副本复制到新的 table 中,并进行一些架构更改?
- 或 "get" 模式,并在 SQL 脚本中编辑它,然后添加一个 table 与该模式?
要复制带有一些架构更改的 table,您必须手动创建和复制:
BEGIN;
CREATE TABLE Customer_new (
[...],
CHECK ([...])
);
INSERT INTO Customer_new SELECT * FROM Customer;
DROP TABLE Customer;
ALTER TABLE Customer_new RENAME TO Customer;
COMMIT;
要读取架构,请在 sqlite3
command-line shell 中执行 .schema Customer
。
这为您提供了 CREATE TABLE 语句,您可以编辑和执行它。
要更改table,您可以使用后门。
首先,阅读实际的 table 定义(这与您从 .schema
中得到的相同):
SELECT sql FROM sqlite_master WHERE type = 'table' AND name = 'Customer';
将您的 CHECK 约束添加到该字符串,然后使用 PRAGMA writable_schema=1; 启用对 sqlite_master
的写入访问权限,并将新的 table 定义写入其中:
UPDATE sqlite_master SET sql='...' WHERE type='table' AND name='Customer';
然后重新打开数据库。
警告:这仅适用于不改变 table 的磁盘格式的更改。如果您确实进行了任何更改记录格式的更改(例如 adding/removing 字段,或修改 rowid,或添加需要内部索引的约束),您的数据库将会崩溃。