我想使用 SQLite FTS3(实际上是 FTS4)来索引具有整数列的表,概念上是这样的:
CREATE VIRTUAL TABLE whole (document INTEGER, page INTEGER, content TEXT,
UNIQUE(document, page)) USING fts4();
我知道 FTS3 将除 rowid 之外的所有列视为 TEXT,因此我将不得不使用两个表:
CREATE VIRTUAL TABLE data USING fts4();
CREATE TABLE metadata(document INTEGER, page INTEGER, UNIQUE(document, page));
我希望能够查询文档或给定文档中的页面:
SELECT DISTINCT document FROM metadata NATURAL JOIN data WHERE content MATCH 'foo';
SELECT page FROM metadata NATURAL JOIN data
WHERE document = 123 AND content MATCH 'foo';
我认为 NATURAL JOIN 要求我确保 rowid 保持同步,但最好的方法是什么?我应该使用 FOREIGN KEY 还是其他约束?子选择会比加入更好吗?
我想插入已经在数据库中的文档和页面以覆盖文本内容。这是否可以通过 SQL 以编程方式进行,还是我必须检查该行是否已存在于 info 表中?
我还想从给定文档的两个表中删除——有没有办法在一个语句中做到这一点?
非常感谢所有建议,但由于我是 SQL 新手,因此特别感谢代码示例!
更新:我完全不清楚如何在这里创建外键约束。如果我选择metadata
作为父表(这是我的偏好,在没有双向约束的情况下):
PRAGMA foreign_keys = ON;
CREATE TABLE metadata (document INTEGER, page INTEGER);
CREATE VIRTUAL TABLE data USING fts4(content TEXT, docid REFERENCES metadata);
我得到Error: vtable constructor failed: data
(不出所料,因为docid
是 的别名rowid
,但我当然不能使用另一列,因为除rowid
must be之外的所有列TEXT
)。
而如果我反过来尝试:
PRAGMA foreign_keys = ON;
CREATE VIRTUAL TABLE data USING fts4();
CREATE TABLE metadata (document INTEGER, page INTEGER, docid REFERENCES data);
表构造成功,但如果我尝试:
INSERT INTO data (docid, content) VALUES (123, 'testing');
INSERT INTO metadata (docid, document, page) VALUES (123, 12, 23);
我明白了Error: foreign key mismatch
。