2

我有这个查询

SELECT a.* 
FROM entries a 
INNER JOIN entries_keywords b ON a.id = b.entry_id 
INNER JOIN keywords c ON b.keyword_id = c.id 
WHERE c.key IN ('wake', 'up') 
GROUP BY a.id 
HAVING COUNT(*) = 2

但它很慢。如何优化设计索引以加快速度?

编辑 这是当前模式

CREATE TABLE `entries` (`id` integer PRIMARY KEY AUTOINCREMENT, `sha` text);
CREATE TABLE `entries_keywords` (`id` integer PRIMARY KEY AUTOINCREMENT, `entry_id` integer REFERENCES `entries`, `keyword_id` integer REFERENCES `keywords`);
CREATE TABLE `keywords` (`id` integer PRIMARY KEY AUTOINCREMENT, `key` string);
CREATE INDEX `entries_keywords_entry_id_index` ON `entries_keywords` (`entry_id`);
CREATE INDEX `entries_keywords_entry_id_keyword_id_index` ON `entries_keywords` (`entry_id`, `keyword_id`);
CREATE INDEX `entries_keywords_keyword_id_index` ON `entries_keywords` (`keyword_id`);
CREATE INDEX `keywords_key_index` ON `keywords` (`key`);

我用的是Sqlite3,查询没有失败,但是很慢。

现在我是这样的查询(每个关键字的子查询):

select *
from (
    select *
    from (entries) e
    inner join entries_keywords ek on e.id = ek.entry_id
    inner join keywords k on ek.keyword_id = k.id
    where k.key = 'wake') e
inner join entries_keywords ek on e.id = ek.entry_id
inner join keywords k on ek.keyword_id = k.id
where k.key = 'up';

这速度更快,但感觉不对,因为如果我有很多关键字,它会变得很难看。

4

1 回答 1

1

该查询所需的关键索引

  • 关键字(键)
  • entry_keywords(keyword_id,entry_id)
  • 条目(ID)

您必须使用 MySQL,否则 SELECT a.* 会失败。
在关于此声明的第二条评论之后编辑,让我指出为什么 select a.*会在这里失败 - 这是因为GROUP BY.

解释一下,因为条件(WHERE)在c.key上,所以需要索引。
然后这会针对 b.keyword_id 进行 JOIN。我们创建一个包含 b.entry_id 的索引,这样它就不必查找表——仅索引就可以覆盖所需的列。
最后,a.id=b.entry_id 连接回条目表,因此我们索引该表的 id。

它很可能entries(id)已经是主键,但您可能已经以相反的方式建立entries_keywords索引- 它无法满足此连接。

于 2012-09-28T21:08:21.260 回答