当前位置:   article > 正文

PostgreSQL GIN索引limit慢的原因分析

pgsql limit 1很慢

PostgreSQL GIN索引的结构如下图 :
假设这个表有2列,一列存储INT,另一列存储INT数组,最左边的表示记录的行号。
756491572128696591
假设对INT数组建立GIN索引,那么GIN索引会记录每个数组element对应的行号,对于行号多的,会存成LIST,然后在索引中指向该list。
669553658485461243
好了接下来分析一下limit慢的原因, 实际上和gin索引的扫描方法有关,目前gin 索引只支持bitmap index scan,也就是说,会将所有匹配的行号取出,排序,然后去heap表取记录。
那么不管你limit多小,根据行号排序是免不了的,这就是limit比btree索引以及gist索引等不需要bitmap index scan的其他索引方法慢的原因。
例子:

  1. postgres=# create table t3(id int, info int[]);
  2. CREATE TABLE
  3. postgres=# insert into t3 select generate_series(1,10000),array[1,2,3,4,5];
  4. INSERT 0 10000
  5. postgres=# create index idx_t3_info on t3 using gin(info);
  6. CREATE INDEX
  7. postgres=# set enable_seqscan=off;
  8. SET

数组匹配,走索引,注意是bitmap index scan,所以被匹配的数组对应有1万条记录的话,这1万条记录的行号会先排序,然后扫描heap取出记录。

  1. postgres=# explain analyze select * from t3 where info && array [1] ;
  2. QUERY PLAN
  3. -----------------------------------------------------------------------------------------------------------------------------
  4. Bitmap Heap Scan on t3 (cost=83.00..302.00 rows=10000 width=45) (actual time=1.156..3.565 rows=10000 loops=1)
  5. Recheck Cond: (info && '{1}'::integer[])
  6. Heap Blocks: exact=94
  7. -> Bitmap Index Scan on idx_t3_info (cost=0.00..80.50 rows=10000 width=0) (actual time=1.129..1.129 rows=10000 loops=1)
  8. Index Cond: (info && '{1}'::integer[])
  9. Planning time: 0.107 ms
  10. Execution time: 5.272 ms
  11. (7 rows)

因为走bitmap index scan, 所以即使加了limit 1,行号排序少不了,开销是不小的。

  1. postgres=# explain analyze select * from t3 where info && array [1] limit 1;
  2. QUERY PLAN
  3. ----------------------------------------------------------------------------------------------------------------------------
  4. Limit (cost=83.00..83.02 rows=1 width=45) (actual time=1.121..1.121 rows=1 loops=1)
  5. -> Bitmap Heap Scan on t3 (cost=83.00..302.00 rows=10000 width=45) (actual time=1.119..1.119 rows=1 loops=1)
  6. Recheck Cond: (info && '{1}'::integer[])
  7. Heap Blocks: exact=1
  8. -> Bitmap Index Scan on idx_t3_info (cost=0.00..80.50 rows=10000 width=0) (actual time=1.095..1.095 rows=10000 loops=1)
  9. Index Cond: (info && '{1}'::integer[])
  10. Planning time: 0.113 ms
  11. Execution time: 1.175 ms
  12. (8 rows)

上面就是gin 索引limit慢的原因。
但是GIN这么设计是有原因的,因为数组中可能存在大量的重复值。
例如我需要找的element有3个1,2,3,假设一共有10万条记录.
而1,2,3对应的ctid中可能存在大量重复的page,那么使用bitmap index scan就可以大大减少离散扫描的情况。
对于获取大量离散存放的堆数据是有奇效的。
而如果获取的记录数比较少,并且数据库的shared buffer足够大的话,完全没有必要bitmap index scan效果一般。

下面扩展一下,另一个例子,使用btree_gin使得一些标准类型也支持GIN索引,因此可以用它来建立联合索引。
联合索引一般用在一个字段选择性不好,但几个字段组合起来选择性就比较好的情况。
例子

  1. postgres=# create extension btree_gin;
  2. CREATE EXTENSION
  3. postgres=# create table t4(id int, info int[]);
  4. CREATE TABLE
  5. postgres=# insert into t4 select trunc(random()*1000), array_append(array[1,2,3], trunc(random()*1000)::int) from generate_series(1,100000);
  6. INSERT 0 100000
  7. postgres=# select * from t4 limit 10;
  8. id | info
  9. -----+-------------
  10. 588 | {1,2,3,835}
  11. 382 | {1,2,3,332}
  12. 817 | {1,2,3,476}
  13. 478 | {1,2,3,597}
  14. 928 | {1,2,3,714}
  15. 645 | {1,2,3,539}
  16. 457 | {1,2,3,536}
  17. 713 | {1,2,3,246}
  18. 842 | {1,2,3,545}
  19. 194 | {1,2,3,70}
  20. (10 rows)
  21. postgres=# create index idx_t4 on t4 using gin(id,info);
  22. CREATE INDEX
  23. postgres=# explain (analyze,verbose,costs,timing,buffers) select * from t4 where id=10 and info && array[1,2,3];
  24. QUERY PLAN
  25. ---------------------------------------------------------------------------------------------------------------------------
  26. Bitmap Heap Scan on public.t4 (cost=10000000010.89..10000000111.71 rows=97 width=44) (actual time=1.572..1.737 rows=97 loops=1)
  27. Output: id, info
  28. Recheck Cond: ((t4.id = 10) AND (t4.info && '{1,2,3}'::integer[]))
  29. Heap Blocks: exact=92
  30. Buffers: shared hit=179
  31. -> Bitmap Index Scan on idx_t4 (cost=0.00..10.87 rows=97 width=0) (actual time=1.554..1.554 rows=97 loops=1)
  32. Index Cond: ((t4.id = 10) AND (t4.info && '{1,2,3}'::integer[]))
  33. Buffers: shared hit=87
  34. Planning time: 0.262 ms
  35. Execution time: 1.786 ms
  36. (10 rows)

gin的联合索引用在什么地方比较好?
使用索引对应字段上的条件可以将范围缩小到很小的场景。
如果不能这样,或者是btree就可以缩小到很小的范围,那么建议使用BTREE就够了。
或者是说使用了limit限制要取的记录数,那么使用btree也是更好的,因为btree可以走index scan也可以走bitmap index scan。适用于小数据量查询,也适用于大数据量查询。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/135245
推荐阅读
相关标签
  

闽ICP备14008679号