Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2951494
  • 博文数量: 199
  • 博客积分: 1400
  • 博客等级: 上尉
  • 技术积分: 4126
  • 用 户 组: 普通用户
  • 注册时间: 2008-07-06 19:06
个人简介

半个PostgreSQL DBA,热衷于数据库相关的技术。我的ppt分享https://pan.baidu.com/s/1eRQsdAa https://github.com/chenhuajun https://chenhuajun.github.io

文章分类

全部博文(199)

文章存档

2020年(5)

2019年(1)

2018年(12)

2017年(23)

2016年(43)

2015年(51)

2014年(27)

2013年(21)

2011年(1)

2010年(4)

2009年(5)

2008年(6)

分类: Mysql/postgreSQL

2015-03-12 15:47:39

2.3 性能推论

根据前面介绍的全文检索的GiST索引的算法,可以对其性能表现有个大体的认识。

1)GiST索引是有损耗的
GiST索引的叶节点中的索引项没有存储实际的keyword值,而只是keyword的Sign值(ARRKEY形式),或者是Sign值的签名文件(SIGNKEY形式),即GiST索引项中存储是有损耗压缩后的key。这意味着该索引可能会产生多余的匹配,所以,对通过索引匹配找到的每个项目需要再从数据文件中抓取元组,进行recheck才能确定是否真的匹配。


2)独特的词汇数越多,SIGNKEY型索引项的准确率越差

这里的准确率指,给定一个key应该匹配的项目数除以实际匹配的项目数。
可以简单的估算出:准确率 = 992/独立的词元个数
手册中建议GiST用于处理100,000以下的独特立词元的场景,即确保准确率大于1/100。

3)SIGNKEY型索引项包含的词元越多,匹配的概率越大
有m个bit位被置成1的SIGNKEY型索引项,匹配任意关键字的概率是m/992。
也就是说对任何一个索引节点,该节点包含的词元集合对应的SIGNKEY中被置为1的bit位数决定了这个索引节点在每次查询时需要被扫描的概率。
SIGNKEY中被置为1的bit位数在词元较少时,近似等于包含的词元数。但随着词元的增加,不同词元对应到同一个bit位的概率会增大,所以索引节点被扫描的概率不是随着包含的词元数线性增长。具体参考下表
包含词元数 SIGNKEY中被置为1的bit位
匹配任意关键字的平均概率
10 10 1%
20 20 2%
50 49 5%
100 95 10%
200 181 18%
500 393 40%
1000 630 64%

上面的表格的数据通过以下公式计算。

点击(此处)折叠或打开

  1. testdb=# with recursive tx1(n,sign_bits) as
  2. (select 1 n,1.0 sign_bits
  3. union all
  4. select n+1, sign_bits + 1 - sign_bits/992 from tx1 where n<=1000
  5. )
  6. select n,sign_bits,sign_bits/992 scan_probability from tx1 where n in(10,20,50,100,200,500,1000);
  7.     n | sign_bits                | scan_probability
  8. ------+--------------------------+------------------------
  9.    10 |   9.95475882520071613263 | 0.01003503913830717352
  10.    20 |  19.80962125597813433986 | 0.01996937626610699026
  11.    50 |  48.78480462616009595302 | 0.04917823046991945157
  12.   100 |  95.17045888698489805581 | 0.09593796258768638917
  13.   200 | 181.21045784981729332855 | 0.18267183250989646505
  14.   500 | 392.89514681946496009988 | 0.39606365606800903236
  15.  1000 | 630.17880533823543738507 | 0.63526089247805991672
  16. (7 rows)

我不太不确定上面公式的正确性,所以又模拟了一下,结果是差不多的。

点击(此处)折叠或打开

  1. testdb=# select count(DISTINCT sign) from (select (random()*992)::int4 sign from generate_series(1,200))aa;
  2.  count
  3. -------
  4.    182
  5. (1 row)

  6. testdb=# select count(DISTINCT sign) from (select (random()*992)::int4 sign from generate_series(1,500))aa;
  7.  count
  8. -------
  9.    397
  10. (1 row)

  11. testdb=# select count(DISTINCT sign) from (select (random()*992)::int4 sign from generate_series(1,1000))aa;
  12.  count
  13. -------
  14.    643
  15. (1 row)

上面的数据假设词元集合的词元是随机的,这对页节点中的SIGNKEY索引项是适用的,但对其他情况就不适用了,因为GiST在组织索引的时候会尽量把SIGNKEY中对应相同Bit位的词元凑在一起。不过,为简化描述,后面会有意无视这个问题。

索引中数量占多数的是叶节点,我们以认点如果含1000个以上的词元就已是一个极限了。这时每次查询平均需要扫描64%以上的叶索引节点,因此索引的效果可能也就值得怀疑了。

那么如何估算点平均含的元数呢?
可以肯定的是一定会大于每个记录包含的平均词元数,也一定大于总的词元数除以总的叶节点数。不妨先得到这两个值,然后乘以某个假想的系数,比如5倍。如果这样估算,只要每个录平均被拆成
100个以上的词元,那么平均每次查询就要扫描超过40%的叶索引节点了。

但是,即使扫描了全部索引节点,由于下面的原因,走索引也有可能比全表扫描快。
a)索引可能比数据小
b)索引元组的匹配计算可能比数据元组的匹配计算快

不过,一旦每个记录包含的平均词元数超过ARRKEY存储形式的阈值(大概120吧),情况又不一样了。
这时所有叶节点中的索引项都是SIGNKEY形式,索引的Size要比数据小很多,但需要对数据文件进行recheck的概率大大增加。对于每个记录包含的平均词元数在200以上,要recheck 超过18%的数据记录,如果这样这个索引基本就报废了。
所以GiST顶多用来索引一下文章标题之类的小段文本,不适合索引整篇文章。

注)关于签名文件索引算法的效率问题,专业的学术论文中应该会有详细的探讨,我在这里,只是做了一下粗浅的分析。

2.4 性能验证

参考我前面一篇博文中使用的测试数据进行简单的性能验证。验证的结果基本和[2.3 性能推论]中的描述是吻合的,以下是详细内容。
http://zzjlzx.blog.chinaunix.net/uid-20726500-id-4824895.html

1)环境
CentOS 6.5
PostgreSQL 9.4.0

2)测试数据
没有特意找什么测试数据,正好手头有一份翻译中的PostgreSQL9.3中文手册,就拿它当测试数据了。

建表

点击(此处)折叠或打开

  1. create table t1(id serial,c1 text);

导数据,手册中的每一行作为一条记录。

点击(此处)折叠或打开

  1. -bash-4.1$ tar xf pg9.3.1.html.tar.gz
  2. -bash-4.1$ find html -name *.html -exec cat {} \;|iconv -f GBK -t UTF8|sed 's\[\]\\g'|psql -p 5433 testdb -c "copy t1(c1) from stdin"
注)翻译中的PostgreSQL9.3中文手册(pg9.3.1.html.tar.gz)可从以下位置下载:

查看数据大小

点击(此处)折叠或打开

  1. testdb=# select count(*) from t1;
  2.  count 
  3. --------
  4.  702476
  5. (row)
  6. testdb=# select pg_table_size('t1');
  7.  pg_table_size 
  8. ---------------
  9.       37519360
  10. (row)

无索引时的查询

点击(此处)折叠或打开

  1. testdb=# \timing
  2. Timing is on.
  3. testdb=# select * from t1 where to_tsvector('testzhcfg',c1) @@ '重排';
  4.    id | c1
  5. --------+----------------------------------------
  6.  561249 | >操作可能因为其它的重排列而被引入。</P
  7. (1 row)

  8. Time: 2739.207 ms
  9. testdb=# select * from t1 where c1 like '%重排%';
  10.    id | c1
  11. --------+----------------------------------------
  12.  561249 | >操作可能因为其它的重排列而被引入。</P
  13. (1 row)

  14. Time: 132.121 ms
可能to_tsvector()的分词处理比较耗时吧,在没有索引的情况下全文检索的匹配比普通的like匹配慢了20倍。

查看总词元数和每记录的平均词元数

点击(此处)折叠或打开

  1. testdb=# select count(*) from (select ts_stat('select to_tsvector(''testzhcfg'', c1) from t1')) aa;
  2.  count
  3. -------
  4.  30645
  5. (1 row)
  6. testdb=# select avg(length(to_tsvector('testzhcfg', c1))) from t1;
  7.         avg
  8. --------------------
  9.  2.9314695448670133
  10. (1 row)
根据之前的假设,GiST索引的准确率为0.032 (0.032=992/30645)。

3)查询的性能测试

GIN索引测试:

点击(此处)折叠或打开

  1. testdb=# drop index t1_c1_idx_gist;
  2. DROP INDEX
  3. Time: 16.144 ms
  4. testdb=# create index t1_c1_idx_gin on t1 using gin(to_tsvector('testzhcfg',c1));
  5. CREATE INDEX
  6. Time: 4463.896 ms
  7. testdb=# select pg_relation_size('t1_c1_idx_gin');
  8.  pg_relation_size 
  9. ------------------
  10.           8765440
  11. (row)

  12. Time: 0.877 ms
  13. testdb=# explain (analyze,buffers) select * from t1 where to_tsvector('testzhcfg',c1) @@ '重排';
  14.                                                         QUERY PLAN 
  15. --------------------------------------------------------------------------------------------------------------------------
  16.  Bitmap Heap Scan on t1 (cost=47.22..4580.24 rows=3512 width=21) (actual time=0.028..0.029 rows=1 loops=1)
  17.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  18.    Heap Blocks: exact=1
  19.    Buffers: shared hit=5
  20.    -> Bitmap Index Scan on t1_c1_idx_gin (cost=0.00..46.35 rows=3512 width=0) (actual time=0.019..0.019 rows=1 loops=1)
  21.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  22.          Buffers: shared hit=4
  23.  Planning time: 0.101 ms
  24.  Execution time: 0.064 ms
  25. (rows)

  26. Time: 0.637 ms
由于匹配的记录数只有一条,所以理想状况下,需要扫描索引页数等于索引的层数,即该记录所在叶索引页加上所有祖先索引页。
GIN只扫描了4页索引,那么GIN索引应该有4层,GIN索引扫描的索引页数是和理想值一致的。


GiST索引测试:

点击(此处)折叠或打开

  1. testdb=# create index t1_c1_idx_gist on t1 using gist(to_tsvector('testzhcfg',c1));
  2. CREATE INDEX
  3. Time: 26349.854 ms
  4. testdb=# select pg_relation_size('t1_c1_idx_gist');
  5.  pg_relation_size 
  6. ------------------
  7.          32841728
  8. (row)

  9. Time: 5.783 ms
  10. testdb=# explain (analyze,buffers) select * from t1 where to_tsvector('testzhcfg',c1) @@ '重排';
  11.                                                          QUERY PLAN 
  12. ----------------------------------------------------------------------------------------------------------------------------
  13.  Bitmap Heap Scan on t1 (cost=111.51..4644.52 rows=3512 width=21) (actual time=3.531..3.532 rows=1 loops=1)
  14.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  15.    Heap Blocks: exact=1
  16.    Buffers: shared hit=220
  17.    -> Bitmap Index Scan on t1_c1_idx_gist (cost=0.00..110.63 rows=3512 width=0) (actual time=3.388..3.388 rows=1 loops=1)
  18.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  19.          Buffers: shared hit=219
  20.  Planning time: 0.314 ms
  21.  Execution time: 3.574 ms
  22. (rows)

  23. Time: 4.861 ms
GiST扫描了219页索引,即扫描了5.5%(5.5%=219/(32841728/8192))的索引页。根据前面的假设反过来推算,每个叶节点平均包含的词元数大概是50的样子。

4)1/5数据量的GiST索引测试:
取之前的原始测试数据的前1/5数据进行测试。

建表并插入数据

点击(此处)折叠或打开

  1. testdb=# create table t2 (like t1);
  2. CREATE TABLE
  3. Time: 33.293 ms
  4. testdb=# insert into t2 select * from t1 limit 702476/5;
  5. INSERT 0 140495
  6. Time: 938.910 ms
  7. testdb=# select count(*) from t2;
  8.  count
  9. --------
  10.  140495
  11. (1 row)

  12. Time: 28.495 ms
  13. testdb=# select pg_relation_size('t2');
  14.  pg_relation_size
  15. ------------------
  16.           7528448
  17. (1 row)

  18. Time: 0.689 ms
  19. testdb=# select * from t2 where to_tsvector('testzhcfg',c1) @@ '验证器';
  20.    id | c1
  21. --------+-----------------
  22.  115726 | >"验证器"</SPAN
  23. (1 row)

  24. Time: 543.530 ms
数据量是原来的t1表的1/5

查看总词元数和每记录的平均词元数

点击(此处)折叠或打开

  1. testdb=# select count(*) from (select ts_stat('select to_tsvector(''testzhcfg''::regconfig, c1) from t2'))aa ;
  2.  count
  3. -------
  4.  13904
  5. (1 row)

  6. Time: 780.354 ms
  7. testdb=# select avg(length(to_tsvector('testzhcfg', c1))) from t2;
  8.         avg
  9. --------------------
  10.  2.9266593117192783
  11. (1 row)

  12. Time: 665.092 ms
总词元数大约是原来的1/2(13904/30645),每记录的平均词元数不变

建GiST索引再查询

点击(此处)折叠或打开

  1. testdb=# create index t2_c1_idx_gist on t2 using gist(to_tsvector('testzhcfg',c1));
  2. CREATE INDEX
  3. Time: 4709.604 ms
  4. testdb=# select pg_relation_size('t2_c1_idx_gist');
  5.  pg_relation_size 
  6. ------------------
  7.           6447104
  8. (row)

  9. Time: 0.431 ms
  10. testdb=# select pg_relation_size('t2_c1_idx_gist')/8192;
  11.  ?column? 
  12. ----------
  13.       787
  14. (row)

  15. Time: 0.675 ms
  16. testdb=# explain (analyze,buffers) select * from t2 where to_tsvector('testzhcfg',c1) @@ '验证器';
  17.                                                         QUERY PLAN 
  18. --------------------------------------------------------------------------------------------------------------------------
  19.  Bitmap Heap Scan on t2 (cost=21.72..931.18 rows=702 width=21) (actual time=1.036..1.037 rows=1 loops=1)
  20.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''验证器'''::tsquery)
  21.    Heap Blocks: exact=1
  22.    Buffers: shared hit=55
  23.    -> Bitmap Index Scan on t2_c1_idx_gist (cost=0.00..21.55 rows=702 width=0) (actual time=0.964..0.964 rows=1 loops=1)
  24.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''验证器'''::tsquery)
  25.          Buffers: shared hit=54
  26.  Planning time: 0.304 ms
  27.  Execution time: 1.123 ms
  28. (rows)

  29. Time: 2.235 ms
GiST扫描了54页索引,即扫描了6.9%(6.9%=54/787)的索引页。根据前面的假设反过来推算,每个叶节点平均包含的词元数大概是70的样子。
索引页比例的上升应该和”总词元数/总索引页数“值的上升有关,之前是7.4(30645/(32841728/8192)) ,现在是17.7(13904/787)

5)增加每条记录的平均词元数后的GiST索引测试:
把原始测试数据的每30条记录合并成一条记录后进行测试。

建表并插入数据

点击(此处)折叠或打开

  1. testdb=# create table t3 (like t1);
  2. CREATE TABLE
  3. Time: 8.205 ms
  4. testdb=# insert into t3 select nid,string_agg(c1,' ') from (select id/30 nid,c1 from t1)aa group by nid;
  5. INSERT 0 23418
  6. Time: 1403.488 ms
  7. testdb=# select count(*) from t3;
  8.  count
  9. -------
  10.  23418
  11. (1 row)

  12. Time: 13.487 ms
  13. testdb=# select pg_relation_size('t3');
  14.  pg_relation_size
  15. ------------------
  16.          13066240
  17. (1 row)

  18. Time: 0.551 ms
  19. testdb=# select * from t3 where c1 like '%重排%';
  20.   id |
  21.                                                                                                                c1
  22.                                                                                                                                                      
  23.                                                                     
  24. -------+---------------------------------------------------------------------------------------------------------------------------------------------
  25. -----------------------------------------------------------------------------------------------------------------------------------------------------
  26. -----------------------------------------------------------------------------------------------------------------------------------------------------
  27. --------------------------------------------------------------------
  28.  18708 | >子句,则为操作符命名一个选择性限制计算函数(注意这里是一个函数名, CLASS="LITERAL" ><P >如果提供了<TT > >NOT (x = y)</TT >RESTRICT</TT CLASS
  29. ="LITERAL" NAME="AEN54157" ><H2 >操作可能因为其它的重排列而被引入。</P >NOT</TT CLASS="LITERAL" CLASS="LITERAL" 这样的表达式简化成<TT >x &lt;&gt; y</
  30. TT >。这样的情况比你想像的要频繁的多, CLASS="SECT2" 因为<TT ><P >否定符对可以用上面交换符对中解释的相同的方法来定义。</P CLASS="SECT2" ><DIV ></DIV
  31. >35.13.3. <TT >RESTRICT</TT ><A ></H2 ></A 而不是一个操作符名)<TT
  32. (1 row)

  33. Time: 36.146 ms

查看总词元数和每记录的平均词元数

点击(此处)折叠或打开

  1. testdb=# select count(*) from (select ts_stat('select to_tsvector(''testzhcfg''::regconfig, c1) from t3'))aa ;
  2.  count
  3. -------
  4.  30634
  5. (1 row)

  6. Time: 3180.648 ms
  7. testdb=# select avg(length(to_tsvector('testzhcfg', c1))) from t3;
  8.          avg
  9. ---------------------
  10.  52.8218464429071654
  11. (1 row)

  12. Time: 2667.861 ms
记录的平均已经从原来的3提高到了53。可以预见叶索引节点包含的平均词元数一定也会大幅度提高。

建GiST索引再查询

点击(此处)折叠或打开

  1. testdb=# create index t3_c1_idx_gist on t3 using gist(to_tsvector('testzhcfg',c1));
  2. CREATE INDEX
  3. Time: 3672.601 ms
  4. testdb=# select pg_relation_size('t3_c1_idx_gist');
  5.  pg_relation_size
  6. ------------------
  7.           7053312
  8. (1 row)

  9. Time: 0.570 ms
  10. testdb=# select pg_relation_size('t3_c1_idx_gist')/8192;
  11.  ?column?
  12. ----------
  13.       861
  14. (1 row)

  15. Time: 0.469 ms
  16. testdb=# analyze ;
  17. ANALYZE
  18. Time: 4000.001 ms
  19. testdb=# explain (analyze,buffers) select * from t3 where to_tsvector('testzhcfg',c1) @@ '重排';
  20.                                                         QUERY PLAN
  21. --------------------------------------------------------------------------------------------------------------------------
  22.  Bitmap Heap Scan on t3 (cost=16.97..305.35 rows=89 width=516) (actual time=15.888..17.608 rows=1 loops=1)
  23.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  24.    Rows Removed by Index Recheck: 14
  25.    Heap Blocks: exact=15
  26.    Buffers: shared hit=183
  27.    -> Bitmap Index Scan on t3_c1_idx_gist (cost=0.00..16.95 rows=89 width=0) (actual time=1.340..1.340 rows=15 loops=1)
  28.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  29.          Buffers: shared hit=168
  30.  Planning time: 0.495 ms
  31.  Execution time: 17.662 ms
  32. (10 rows)

  33. Time: 20.008 ms
GiST扫描了168页索引,即扫描了19.5%(19.5%=168/861)的索引页。根据前面的假设反过来推算,每个叶节点平均包含的词元数大概是200的样子。
需要注意的是,Recheck的记录数增加了。要Recheck 15条记录,而查询的大部分时间花在Recheck上。

6)增加每条记录的平均词元数后的GiST索引测试2:
把原始测试数据的每300条记录合并成一条记录后进行测试。

建表并插入数据

点击(此处)折叠或打开

  1. testdb=# create table t4 (like t1);
  2. CREATE TABLE
  3. Time: 10.509 ms
  4. testdb=# insert into t4 select nid,string_agg(c1,' ') from (select id/300 nid,c1 from t1)aa group by nid;
  5. INSERT 0 2344
  6. Time: 1167.627 ms
  7. testdb=# select count(*) from t4;
  8.  count
  9. -------
  10.   2344
  11. (1 row)

  12. Time: 1.203 ms
  13. testdb=# select pg_relation_size('t4');
  14.  pg_relation_size
  15. ------------------
  16.            950272
  17. (1 row)

  18. Time: 0.540 ms
  19. testdb=# select id,length(c1) from t4 where c1 like '%重排%';
  20.   id | length
  21. ------+--------
  22.  1870 | 4489
  23. (1 row)

  24. Time: 87.053 ms
  25. testdb=# select avg(length(c1)) from t4;
  26.           avg
  27. -----------------------
  28.  4358.0912969283276451
  29. (1 row)


  30. Time: 126.893 ms
每个文本字段的平均大小为4000多个字符,有点像一般文章的正文大小了。

查看总词元数和每记录的平均词元数

点击(此处)折叠或打开

  1. testdb=# select count(*) from (select ts_stat('select to_tsvector(''testzhcfg''::regconfig, c1) from t4'))aa ;
  2.  count
  3. -------
  4.  30633
  5. (1 row)

  6. Time: 3213.285 ms
  7. testdb=# select avg(length(to_tsvector('testzhcfg', c1))) from t4;
  8.          avg
  9. ----------------------
  10.  288.1040955631399317
  11. (1 row)

  12. Time: 2890.361 ms
记录的平均已经从原来的3提高到了288。因此叶索引节点的索引项基本都是SIGNKEY形式的存储了。

建GiST索引再查询

点击(此处)折叠或打开

  1. testdb=# create index t4_c1_idx_gist on t4 using gist(to_tsvector('testzhcfg',c1));
  2. CREATE INDEX
  3. Time: 3026.344 ms
  4. testdb=# select pg_relation_size('t4_c1_idx_gist');
  5.  pg_relation_size
  6. ------------------
  7.            499712
  8. (1 row)

  9. Time: 0.605 ms
  10. testdb=# select pg_relation_size('t4_c1_idx_gist')/8192;
  11.  ?column?
  12. ----------
  13.        61
  14. (1 row)

  15. Time: 0.438 ms
  16. testdb=# analyze;
  17. ANALYZE
  18. Time: 6775.215 ms
  19. testdb=# explain (analyze,buffers) select * from t4 where to_tsvector('testzhcfg',c1) @@ '重排';
  20.                                                         QUERY PLAN
  21. --------------------------------------------------------------------------------------------------------------------------
  22.  Bitmap Heap Scan on t4 (cost=4.24..40.84 rows=12 width=361) (actual time=164.155..187.033 rows=1 loops=1)
  23.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  24.    Rows Removed by Index Recheck: 134
  25.    Heap Blocks: exact=62
  26.    Buffers: shared hit=523
  27.    -> Bitmap Index Scan on t4_c1_idx_gist (cost=0.00..4.24 rows=12 width=0) (actual time=0.379..0.379 rows=135 loops=1)
  28.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''重排'''::tsquery)
  29.          Buffers: shared hit=53
  30.  Planning time: 0.341 ms
  31.  Execution time: 187.089 ms
  32. (10 rows)

  33. Time: 187.954 ms
GiST扫描了53页索引,即扫描了86.9%(86.9%=53/61)的索引页。根据前面的假设反过来推算,每个叶节点平均包含的词元数应该在2000以上。

现在需要Recheck的记录数增加到了135,Recheck花了太多的时间,以至于性能还不如用like进行全表扫描(87.053 ms)。
Recheck的记录数占总记录数的5.8%(5.8%=135/2344)。但是按照概率去算,每个记录包含288个词元的话,平均需要Recheck的记录数应该要占到20%。于是
又尝试了几个关键字,有大于20%的,也有小于20%的。看来是因为记录在SIGNKEY的bit位上的分布不均匀导致的差异,平均值仍然可以认为是20% 。

点击(此处)折叠或打开

  1. testdb=# explain (analyze,buffers) select * from t4 where to_tsvector('testzhcfg',c1) @@ 'A';
  2.                                                         QUERY PLAN
  3. ---------------------------------------------------------------------------------------------------------------------------
  4.  Bitmap Heap Scan on t4 (cost=4.24..40.84 rows=12 width=361) (actual time=1709.532..1709.532 rows=0 loops=1)
  5.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''A'''::tsquery)
  6.    Rows Removed by Index Recheck: 1395
  7.    Heap Blocks: exact=110
  8.    Buffers: shared hit=3962
  9.    -> Bitmap Index Scan on t4_c1_idx_gist (cost=0.00..4.24 rows=12 width=0) (actual time=0.659..0.659 rows=1395 loops=1)
  10.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''A'''::tsquery)
  11.          Buffers: shared hit=61
  12.  Planning time: 0.250 ms
  13.  Execution time: 1709.586 ms
  14. (10 rows)

  15. Time: 1710.378 ms
  16. testdb=# explain (analyze,buffers) select * from t4 where to_tsvector('testzhcfg',c1) @@ 'B';
  17.                                                         QUERY PLAN
  18. ---------------------------------------------------------------------------------------------------------------------------
  19.  Bitmap Heap Scan on t4 (cost=4.24..40.84 rows=12 width=361) (actual time=1877.729..1877.729 rows=0 loops=1)
  20.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''B'''::tsquery)
  21.    Rows Removed by Index Recheck: 1127
  22.    Heap Blocks: exact=114
  23.    Buffers: shared hit=3076
  24.    -> Bitmap Index Scan on t4_c1_idx_gist (cost=0.00..4.24 rows=12 width=0) (actual time=0.462..0.462 rows=1127 loops=1)
  25.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''B'''::tsquery)
  26.          Buffers: shared hit=61
  27.  Planning time: 0.208 ms
  28.  Execution time: 1877.770 ms
  29. (10 rows)

  30. Time: 1878.462 ms
  31. testdb=# explain (analyze,buffers) select * from t4 where to_tsvector('testzhcfg',c1) @@ 'C';
  32.                                                         QUERY PLAN
  33. --------------------------------------------------------------------------------------------------------------------------
  34.  Bitmap Heap Scan on t4 (cost=4.24..40.84 rows=12 width=361) (actual time=538.436..538.436 rows=0 loops=1)
  35.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''C'''::tsquery)
  36.    Rows Removed by Index Recheck: 342
  37.    Heap Blocks: exact=99
  38.    Buffers: shared hit=1071
  39.    -> Bitmap Index Scan on t4_c1_idx_gist (cost=0.00..4.24 rows=12 width=0) (actual time=0.382..0.382 rows=342 loops=1)
  40.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''C'''::tsquery)
  41.          Buffers: shared hit=60
  42.  Planning time: 0.312 ms
  43.  Execution time: 538.479 ms
  44. (10 rows)

  45. Time: 539.243 ms
  46. testdb=# explain (analyze,buffers) select * from t4 where to_tsvector('testzhcfg',c1) @@ 'D';
  47.                                                         QUERY PLAN
  48. --------------------------------------------------------------------------------------------------------------------------
  49.  Bitmap Heap Scan on t4 (cost=4.24..40.84 rows=12 width=361) (actual time=445.309..445.309 rows=0 loops=1)
  50.    Recheck Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''D'''::tsquery)
  51.    Rows Removed by Index Recheck: 331
  52.    Heap Blocks: exact=95
  53.    Buffers: shared hit=1139
  54.    -> Bitmap Index Scan on t4_c1_idx_gist (cost=0.00..4.24 rows=12 width=0) (actual time=0.369..0.369 rows=331 loops=1)
  55.          Index Cond: (to_tsvector('testzhcfg'::regconfig, c1) @@ '''D'''::tsquery)
  56.          Buffers: shared hit=59
  57.  Planning time: 0.207 ms
  58.  Execution time: 445.350 ms
  59. (10 rows)

  60. Time: 445.993 ms



阅读(3188) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~