11

我想使用 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,但我当然不能使用另一列,因为除rowidmust 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

4

2 回答 2

7

简而言之,在涉及 FTS3 的情况下很难实现一致性。

SQLite 虚拟表不允许触发器,FTS3 表忽略约束和关联。

到目前为止,我能做的最好的事情如下:

CREATE TABLE metadata (document INTEGER, page INTEGER, UNIQUE(document, page));
CREATE VIRTUAL TABLE data USING fts4();

CREATE VIEW whole AS SELECT metadata.rowid AS rowid, document, page, content 
    FROM metadata JOIN data ON metadata.rowid = data.rowid;

CREATE TRIGGER whole_insert INSTEAD OF INSERT ON whole
BEGIN
  INSERT INTO metadata (document, page) VALUES (NEW.document, NEW.page);
  INSERT INTO data (rowid, content) VALUES (last_insert_rowid(), NEW.content);
END;

CREATE TRIGGER whole_delete INSTEAD OF DELETE ON whole
BEGIN
  DELETE FROM metadata WHERE rowid = OLD.rowid;
  DELETE FROM data WHERE rowid = OLD.rowid;
END;

为了强制保持一致性,我可以(使用)创建触发器来引发对and表PRAGMA recursive_triggers = NO的直接操作的异常,但这对于我的目的来说可能是多余的(同样,我不需要表的触发器)。metadatadataUPDATEwhole

于 2011-07-18T18:58:58.967 回答
-1

我认为大多数 DBA 都会同意,如果您使用 SQL,则应该利用它提供的所有功能。

外键是我推荐的一般路线,它们在此处记录。

一般来说,对于 SQL 数据库,您永远不应该手动强制执行一致性。特别是在与这样的外键完美匹配的情况下。

对于 DELETE FROM 的情况,SQLite 不支持像 MS SQL 这样的“级联”关键字,但它有触发器让你无论如何都有这种行为。SQLite 触发器的文档可以在这里找到。

最后,我会跳过自然连接。

于 2011-07-16T08:16:25.540 回答