Postgresql执行计划专题

1. 执行计划的解释

1.1 EXPLAIN命令

PostgreSQL中EXPLAIN命令的语法格式如下:

EXPLAIN [ ( option [, ...] ) ] statement
EXPLAIN [ ANALYZE ] [ VERBOSE ] statement

该命令的可选项“options”如下:

ANALYZE [ boolean ]
VERBOSE [ boolean ]
COSTS [ boolean ]
BUFFERS [ boolean ]
FORMAT { TEXT | XML | JSON | YAML }
ANALYZE选项

ANALYZE选项通过实际执行SQL来获得SQL命令的实际执行计划。
ANALYZE选项查看到的执行计划因为真正被执行过, 所以可以看到执行计划每一步耗费了多长时间, 以及它实际返回的行数。

加上ANALYZE选项后是真正执行实际的SQL命令, 如果SQL语句是一个插入、删除、 更新或CREATE TABLE AS语句(这些语句会修改数据库) , 为了不影响实际数据, 可以把EXPLAIN ANALYZE放到一个事务中, 执行完后即回滚事务, 命令如下:

BEGIN;
EXPLAIN ANALYZE ...;
ROLLBACK;
VERBOSE选项

VERBOSE选项显示计划的附加信息, 如计划树中每个节点输出的各个列, 如果触发器被触发, 还会输出触发器的名称。
该选项的值默认为“FALSE”。

COSTS选项

COSTS选项显示每个计划节点的启动成本和总成本, 以及估计行数和每行宽度。
该选项的值默认为“TRUE”。

BUFFERS选项

BUFFERS选项显示缓冲区使用的信息。
该参数只能与ANALYZE参数一起使用。
显示的缓冲区信息包括共享块读和写的块数、 本地块读和写的块数, 以及临时块读和写的块数。
共享块、 本地块和临时块分别包含表和索引、 临时表和临时索引, 以及在排序和物化计划中使用的磁盘块。
上层节点显示出来的块数包括所有其子节点使用的块数。
该选项的值默认为“FALSE”。

FORMAT选项

FORMAT选项指定输出格式, 输出格式可以是TEXT、 XML、 JSON或者YAML。
非文本输出包含与文本输出格式相同的信息, 但其他程序更易于解析。
该参数默认为“TEXT”。其他格式见1.3 EXPLAIN使用示例。

1.2 EXPLAIN输出结果解释

下面以一个简单的EXPLAIN的输出结果做解释:

osdba=# explain select * from testtab01;
                           QUERY PLAN
---------------------------------------------------------------
  Seq Scan on testtab01 (cost=0.00..184.00 rows=10000 width=36)
(1 row)

上面的运行结果中“Seq Scan on testtab01”表示顺序扫描表“testtab01”, 顺序扫描也就是全表扫描, 即从头到尾地扫描表。
后面的内容“(cost=0.00..184.00 rows=10000 width=36)”可以分为以下3个部分:

  • cost=0.00..184.00: “cost=”后面有两个数字, 中间由“..”分隔, 第一个数字“0.00”表示启动的成本, 也就是说, 返回第一行需要多少cost值; 第二个数字表示返回所有数据的成本, 关于成本“cost”后面会解释。
  • rows=10000: 表示会返回10000行。
  • width=36: 表示每行平均宽度为36字节。

成本“cost”用于描述SQL命令的执行代价, 默认情况下, 不同操作的cost值如下:

  • 顺序扫描一个数据块, cost值定为“1”。
  • 随机扫描一个数据块, cost值定为“4”。
  • 处理一个数据行的CPU代价, cost值定为“0.01”。
  • 处理一个索引行的CPU代价, cost值定为“0.005”。
  • 每个操作符的CPU代价为“0.0025”。

根据上面的操作类型, PostgreSQL可以智能地计算出一个SQL命令的执行代价, 虽然计算结果不是很精确, 但大多数情况下够用了。

更复杂的执行计划如下:

osdba=# explain select a.id,b.note from testtab01 a,testtab02 b where a.id=b.id;
QUERY PLAN
-----------------------------------------------------------------------------
Hash Join (cost=309.00..701.57 rows=9102 width=36)
Hash Cond: (b.id = a.id)
-> Seq Scan on testtab02 b (cost=0.00..165.02 rows=9102 width=36)
-> Hash (cost=184.00..184.00 rows=10000 width=4)
-> Seq Scan on testtab01 a (cost=0.00..184.00 rows=10000 width=4)
(5 rows)

除“Seq Scan”全表扫描外, 还有一些其他的操作, 如“Hash”“Hash Join”等, 这些内容见下文。

1.3 EXPLAIN使用示例

默认情况下输出的执行计划是文本格式, 但也可以输出JSON格式, 示例如下:

osdba=# explain (format json) select * from testtab01;
QUERY PLAN
----------------------------------------
[ +
{ +
"Plan": { +
"Node Type": "Seq Scan", +
"Relation Name": "testtab01",+
"Alias": "testtab01", +
"Startup Cost": 0.00, +
"Total Cost": 184.00, +
"Plan Rows": 10000, +
"Plan Width": 36 +
} +
} +
]
(1 row)

也可以输出XML格式, 示例如下:

osdba=# explain (format xml) select * from testtab01;
QUERY PLAN
----------------------------------------------------------
<explain xmlns="http://www.postgresql.org/2009/explain">+
<Query> +
<Plan> +
<Node-Type>Seq Scan</Node-Type> +
<Relation-Name>testtab01</Relation-Name> +
<Alias>testtab01</Alias> +
<Startup-Cost>0.00</Startup-Cost> +
<Total-Cost>184.00</Total-Cost> +
<Plan-Rows>10000</Plan-Rows> +
<Plan-Width>36</Plan-Width> +
</Plan> +
</Query> +
</explain>
(1 row)

还可以输出YAML格式, 示例如下:

osdba=# explain (format YAML ) select * from testtab01;
QUERY PLAN
---------------------------------
- Plan: +
Node Type: "Seq Scan" +
Relation Name: "testtab01" +
Alias: "testtab01" +
Startup Cost: 0.00 +
Total Cost: 184.00 +
Plan Rows: 10000 +
Plan Width: 36
(1 row)

添加“analyze”参数, 通过实际执行来获得更精确的执行计划, 命令如下:

osdba=# explain analyze select * from testtab01;
QUERY PLAN--------------------------------------------------------------------------------
Seq Scan on testtab01 (cost=0.00..184.00 rows=10000 width=36) (actual time=0.493..4.320 rows=10000 loops
=1)
Total runtime: 5.653 ms
(2 rows)

从上面的运行结果中可以看出, 加了“analyze”参数后, 可以看到实际的启动时间(第一行返回的时间) 、 执行时间、 实际的扫描行数(actual time=0.493..4.320 rows=10000 loops=1) , 其中启动时间为0.493毫秒, 返回所有行的时间为4.320毫秒, 返回的行数是10000。

analyze选项还有另一种语法, 即放在小括号内, 得到的结果与上面的结果完全一致, 示例如下:

osdba=# explain (analyze true) select * from testtab01;
QUERY PLAN
--------------------------------------------------------------------------------
Seq Scan on testtab01 (cost=0.00..184.00 rows=10000 width=36) (actual time=0.019..2.650 rows=10000 loo
ps=1)
Total runtime: 4.004 ms
(2 rows)

如果只查看执行的路径情况而不看cost值, 则可以加“(costs false)”选项, 命令如下:

osdba=# explain (costs false) select * from testtab01;
QUERY PLAN
-----------------------
Seq Scan on testtab01
(1 row)

联合使用analyze选项和buffers选项, 通过实际执行来查看实际的代价和缓冲区命中的情况, 命令如下:

osdba=# explain (analyze true,buffers true ) select * from testtab03;
QUERY PLAN
--------------------------------------------------------------------------------
Seq Scan on testtab03 (cost=0.00..474468.18 rows=26170218 width=36) (actual time=0.498..8543.701 rows=
10000000 loops=1)
Buffers: shared hit=16284 read=196482 written=196450
Total runtime: 9444.707 ms
(3 rows)

因为加了buffers选项, 执行计划的结果中就会出现一行“Buffers:shared hit=16284 read=196482 written=196450”, 其中“shared hit=16284”表示在共享内存中直接读到16284个块, 从磁盘中读到196482块, 写磁盘196450块。 有人可能会问, SELECT为什么会写? 这是因为共享内存中有脏块, 从磁盘中读出的块必须把内存中的脏块挤出内存, 所以产生了很多的写。

来看下面这个“create table as”的执行计划:

osdba=# explain create table testtab04 as select * from testtab03 limit 100000;
QUERY PLAN
-----------------------------------------------------------------------------
Limit (cost=0.00..3127.66 rows=100000 width=142)
-> Seq Scan on testtab03 (cost=0.00..312766.02 rows=10000002 width=142)
(2 rows)

看一下insert语句的执行计划:

osdba=# explain insert into testtab04 select * from testtab03 limit 100000;
QUERY PLAN
--------------------------------------------------------------------------------
Insert on testtab04 (cost=0.00..4127.66 rows=100000 width=142)
-> Limit (cost=0.00..3127.66 rows=100000 width=142)
-> Seq Scan on testtab03 (cost=0.00..312766.02 rows=10000002 width=142)
(3 rows)

删除语句的执行计划如下:

osdba=# explain delete from testtab04;
QUERY PLAN
-------------------------------------------------------------------
Delete on testtab04 (cost=0.00..22.30 rows=1230 width=6)
-> Seq Scan on testtab04 (cost=0.00..22.30 rows=1230 width=6)
(2 rows)

更新语句的执行计划如下:

osdba=# explain update testtab04 set note='bbbbbbbbbbbbbbbb';
QUERY PLAN
--------------------------------------------------------------------
Update on testtab04 (cost=0.00..22.30 rows=1230 width=10)
-> Seq Scan on testtab04 (cost=0.00..22.30 rows=1230 width=10)
(2 rows)

1.4 全表扫描

全表扫描在PostgreSQL中也称顺序扫描(Seq Scan) , 全表扫描就是把表中的所有数据块从头到尾读一遍, 然后从中找到符合条件的数据块。

全表扫描在EXPLAIN命令的输出结果中用“Seq Scan”表示, 示例如下:

osdba=# EXPLAIN SELECT * FROM testtab01;
QUERY PLAN
---------------------------------------------------------------
Seq Scan on testtab01 (cost=0.00..2754.05 rows=151905 width=36)
(1 row)

1.5 索引扫描

索引通常是为了加快查询数据的速度而增加的。
索引扫描, 就是在索引中找出需要的数据行的物理位置, 然后再到表的数据块中把相应的数据读出来的过程。

索引扫描在EXPLAIN命令的输出结果中用“Index Scan”表示, 示例如下:

osdba=# EXPLAIN SELECT * FROM testtab01 where id=1000;
QUERY PLAN
--------------------------------------------------------------------------------
Index Scan using idx_testtab01_id on testtab01 (cost=0.29..8.31 rows=1 width=70)
Index Cond: (id = 1000)
(2 rows)

1.6 位图扫描

位图扫描也是走索引的一种方式。
方法是扫描索引, 把满足条件的行或块在内存中建一个位图, 扫描完索引后, 再根据位图到表的数据文件中把相应的数据读出来。

如果走了两个索引, 可以把两个索引形成的位图通过AND或OR计算合并成一个, 再到表的数据文件中把数据读出来。

当执行计划的结果行数很多时会走这种扫描, 如非等值查询、 IN子句或有多个条件都可以走不同的索引时。

下面是非等值的一个示例:

osdba=# explain select * from testtab02 where id2 >10000;
QUERY PLAN
--------------------------------------------------------------------------------
Bitmap Heap Scan on testtab02 (cost=18708.13..36596.06 rows=998155 width=16)
Recheck Cond: (id2 > 10000)
-> Bitmap Index Scan on idx_testtab02_id2 (cost=0.00..18458.59 rows=998155 width=0)
Index Cond: (id2 > 10000)
(4 rows)

在位图扫描中可以看到, “Bitmap Index Scan”先在索引中找到符合条件的行, 然后在内存中创建位图, 再到表中扫描, 也就是我们看到的“Bitmap Heap Scan”。
还会看到“Recheck Cond:(id2>10000)”, 这是因为多版本的原因, 从索引中找出
的行从表中读出后还需要再检查一下条件。

下面是一个因为IN子句走位图索引的示例:

osdba=# explain select * from testtab02 where id1 in (2,4,6,8);
QUERY PLAN
---------------------------------------------------------------------------------
Bitmap Heap Scan on testtab02 (cost=17.73..33.47 rows=4 width=16)
Recheck Cond: (id1 = ANY ('{2,4,6,8}'::integer[]))
-> Bitmap Index Scan on idx_testtab02_id1 (cost=0.00..17.73 rows=4 width=0)
Index Cond: (id1 = ANY ('{2,4,6,8}'::integer[]))
(4 rows)

下面是走两个索引后将位图进行BitmapOr运算的示例:

osdba=# explain select * from testtab02 where id2 >10000 or id1 <200000;
QUERY PLAN
----------------------------------------------------------------------------------
Bitmap Heap Scan on testtab02 (cost=20854.46..41280.46 rows=998446 width=16)
Recheck Cond: ((id2 > 10000) OR (id1 < 200000))
-> BitmapOr (cost=20854.46..20854.46 rows=1001000 width=0)
-> Bitmap Index Scan on idx_testtab02_id2 (cost=0.00..18458.59 rows=998155 width=0)
Index Cond: (id2 > 10000)
-> Bitmap Index Scan on idx_testtab02_id1 (cost=0.00..1896.65 rows=102430 width=0)
Index Cond: (id1 < 200000)
(7 rows)

在上面的执行计划中, 可以看到BitmapOr操作, 即使用OR运算合并两个位图。

1.7 条件过滤

条件过滤, 一般就是在WHERE子句上加过滤条件, 当扫描数据行时会找出满足过滤条件的行。
条件过滤在执行计划中显示为“Filter”, 示例如下:

osdba=# EXPLAIN SELECT * FROM testtab01 where id<1000 and note like 'asdk%';
QUERY PLAN
--------------------------------------------------------------------------------
Index Scan using idx_testtab01_id on testtab01 (cost=0.29..48.11 rows=1 width=70)
Index Cond: (id < 1000)
Filter: (note ~~ 'asdk%'::text)

如果条件的列上有索引, 可能会走索引而不走过滤, 示例如下:

osdba=# EXPLAIN SELECT * FROM testtab01 where id<1000;
QUERY PLAN
--------------------------------------------------------------------------------
Index Scan using idx_testtab01_id on testtab01 (cost=0.29..45.63 rows=991 width=70)
Index Cond: (id < 1000)
(2 rows)

osdba=# EXPLAIN SELECT * FROM testtab01 where id>1000;
QUERY PLAN
-----------------------------------------------------------------
Seq Scan on testtab01 (cost=0.00..2485.00 rows=99009 width=70)
Filter: (id > 1000)
(2 rows)

1.8 嵌套循环连接

嵌套循环连接(NestLoop Join) 是在两个表做连接时最朴素的一种连接方式。
在嵌套循环中, 内表被外表驱动, 外表返回的每一行都要在内表中检索找到与它匹配的行, 因此整个查询返回的结果集不能太大(大于1万不适合) , 要把返回子集较小的表作为外表,而且在内表的连接字段上要有索引, 否则速度会很慢。

执行的过程如下: 确定一个驱动表(Outer Table) , 另一个表为Inner Table, 驱动表中的每一行与Inner Table表中的相应记录Join类似一个嵌套的循环。

适用于驱动表的记录集比较?。?lt;10000) 而且Inner Table表有有效的访问方法(Index) 。
需要注意的是, Join的顺序很重要, 驱动表的记录集一定要小, 返回结果集的响应时间才是最快的。

1.9 散列连接

优化器使用两个表中较小的表, 利用连接键在内存中建立散列表, 然后扫描较大的表并探测散列表, 找出与散列表匹配的行。

这种方式适用于较小的表可以完全放于内存中的情况, 这样总成本就是访问两个表的成本之和。
但是如果表很大, 不能完全放入内存, 优化器会将它分割成若干不同的分区,把不能放入内存的部分写入磁盘的临时段, 此时要有较大的临时段从而尽量提高I/O的性能。

下面就是一个散列连接(Hash Join) 的例子:

osdba=# explain select a.id,b.id,a.note from testtab01 a, testtab02 b where a.id=b.id and b.id<=1000000;
QUERY PLAN
--------------------------------------------------------------------------------
Hash Join (cost=20000041250.75..20000676975.71 rows=999900 width=93)
Hash Cond: (a.id = b.id)
-> Seq Scan on testtab01 a (cost=10000000000.00..10000253847.55 rows=10000055 width=89)
-> Hash (cost=10000024846.00..10000024846.00 rows=999900 width=4)
-> Seq Scan on testtab02 b (cost=10000000000.00..10000024846.00 rows=999900 width=4)
Filter: (id <= 1000000)
(6 rows)

先看表大小, 命令如下:

osdba=# select pg_relation_size('testtab01');
pg_relation_size
------------------
1260314624
(1 row)

osdba=# select pg_relation_size('testtab02');
pg_relation_size
------------------
101138432
(1 row)

因为表“'testtab01”大于“'testtab02”, 所以Hash Join是先在较小的表“testtab02”上建立散列表, 然后扫描较大的表“testtab01”并探测散列表, 找出与散列表匹配的行。

1.10 合并连接

通常情况下, 散列连接的效果比合并连接要好, 然而如果源数据上有索引, 或者结果已经被排过序, 此时执行排序合并连接不需要再进行排序, 合并连接的性能会优于散列连接。

下面的示例中, 表“testtab01”的“id”字段上有索引, 表“testtab02”的“id”字段上也有索引, 这时从索引扫描的数据已经排好序了, 就可以直接进行合并连接(Merge Join) :

osdba=# explain select a.id,b.id,a.note from testtab01 a, testtab02 b where a.id=b.id and b.id<=100000;
QUERY PLAN
--------------------------------------------------------------------------------
Merge Join (cost=1.47..47922.57 rows=99040 width=93)
Merge Cond: (a.id = b.id)
-> Index Scan using idx_testtab01_id on testtab01 a (cost=0.43..413538.43 rows=10000000 width=89)
-> Index Only Scan using idx_testtab02_id on testtab02 b (cost=0.42..4047.63 rows=99040 width=4)
Index Cond: (id <= 100000)
(5 rows)

把表“testtab02”上的索引删除, 下面的示例中的执行计划是把testtab02排序后再走Merge Join:

osdba=# drop index idx_testtab02_id;
DROP INDEX

osdba=# explain select a.id,b.id,a.note from testtab01 a, testtab02 b where a.id=b.id and b.id<=100000;
QUERY PLAN
--------------------------------------------------------------------------------
Merge Join (cost=34419.21..78788.84 rows=99040 width=93)
Merge Cond: (a.id = b.id)
-> Index Scan using idx_testtab01_id on testtab01 a (cost=0.43..413538.43 rows=10000000 width=89)
-> Materialize (cost=34418.70..34913.90 rows=99040 width=4)
-> Sort (cost=34418.70..34666.30 rows=99040 width=4)
Sort Key: b.id
-> Seq Scan on testtab02 b (cost=0.00..24846.00 rows=99040 width=4)
Filter: (id <= 100000)
(8 rows)

从上面的执行计划中可以看到“Sort Key: b.id”, 就是对表“testtab02”的“id”字段进行排序。

2. 与执行计划相关的配置项

2.1 ENABLE_*参数

在PostgreSQL中有一些以“ENABLE_”开头的参数, 这些参数提供了影响查询优化器选择不同执行计划的方法。
有时, 如果优化器为特定查询选择的执行计划并不是最优的,可以设置这些参数强制优化器选择一个更好的执行计划来临时解决这个问题。

一般不会在PostgreSQL中配置来改变这些参数值的默认值, 因为通常情况下, PostgreSQL不会走错执行计划。
PostgreSQL走错执行计划是统计信息收集得不及时导致的, 可通过更频繁地运行ANALYZE来解决这个问题, 使用“ENABLE_”只是一个临时的解决方法。

这些参数的详细说明见下表:

参数名称 类型 说明
enable_seqscan boolean 是否选择全表顺序扫描。实际上,并不能完全禁止全表扫描,但是把该变量关闭会让优化器在存在其他方法时优先选择其他方法。
enable_indexscan boolean 是否选择索引扫描。
enable_bitmapscan boolean 是否选择位图扫描。
enable_tidscan boolean (此处说明可能有误或缺失,因为与“是否选择位图扫描”重复,且后文提到多表连接,可能是对另一个参数的描述混淆)注意:原文本中此处的说明与参数名不匹配,根据上下文,此参数可能与特定类型的扫描有关,但具体说明缺失或错误。
enable_nestloop boolean 多表连接时,是否选择嵌套循环连接。如果设置为“off”(注意原文本中写的是“of”,但应为“off”),执行计划只有走嵌套循环连接一条路时,优化器也只能选择这条路,但如果有其他连接方法可以选择,优化器会优先选择其他方法。
enable_hashjoin boolean 多表连接时,是否选择Hash连接。
enable_mergejoin boolean 多表连接时,是否选择Merge连接。
enable_hashagg boolean 是否使用Hash聚合。
enable_sort boolean 是否使用明确的排序。如果设置为“off”(同样注意原文本中的“of”应为“off”),执行计划只有排序一条路时,优化器也只能选择这条路,但如果有其他方法可以选择,优化器会优先选择其他方法。

2.2 COST基准值参数

执行计划在选择最优路径时, 不同路径的cost值只有相对意义, 同时缩放它们将不会对不同路径的选择产生任何影响。

默认情况下, 它们以顺序扫描一个数据块的开销作为基准单位, 也就是说, 将顺序扫描的基准参数“seq_page_cost”默认设为“1.0”, 其他开销的基准参数都对照它来设置。

从理论上来说也可以使用其他基准方法, 如以毫秒计的实际执行时间作基准, 但这些基准方法可能会更复杂一些。

这些COST基准值参数如表:

参数名称 类型 说明
seq_page_cost float 执行计划中一次顺序访问一个数据块页面的开销。默认值是“1.0”
random_page_cost float 执行计划中计算随机访问一个数据块页面的开销。默认值是“4.0”,也就是说,随机访问一个数据块页的开销是顺序访问的开销的4倍
cpu_tuple_cost float 执行计划中计算处理一条数据行的开销。默认值为“0.01”
cpu_index_tuple_cost float 执行计划中计算处理一条索引行的开销。默认值为“0.005” (注意:原图中此参数在cpu_operator_cost之后被提及,但为保持表格整齐,我将其插入到了适当位置)
cpu_operator_cost float 执行计划中执行一个操作符或函数的开销。默认值为“0.025” (原图中写为0.0025,但根据上下文和常见设置,我推测这里应为0.025,且后文提到的effective_cache_size说明中提到了该参数的默认值,与此处相符)
effective_cache_size int 执行计划中在一次索引扫描中可用的磁盘缓冲区的有效大小。更高的数值会导致更可能使用索引扫描,更低的数值会导致更有可能选择顺序全表扫描。该参数对PostgreSQL分配的共享内存大小没有任何影响,它只用于执行计划中代价的估算。数值是用数据页来计算的,通常每个页面大小是8KB。默认是16384个数据块大小,即128MB

在上面的配置项中, “seq_page_cost”一般作为基准, 不用改变。

可能需要改变的是“random_page_cost”, 如果在读数据时, 数据基本都命中在内存中, 这时随机读和顺序读的差异不大, 可能需要把“random_page_cost”的值调得小一些。

如果想让优化器偏向走索引, 而不走全表扫描, 可以把“random_page_cost”的值调得低一些。

2.3 基因查询优化的参数

GEQO是一个使用探索式搜索来执行查询规划的算法, 它可以缩短负载查询的规划时间。
GEQO的检索是随机的, 因此它生成的执行计划会有不可确定性。

基因查询优化器的相关的配置参数如下表:

参数名称 类型 说明
geqo boolean 允许或禁正基因查询优化。在生产系统中建议把此参数打开,默认是打开的。
geqo_threshold integer 提供了一种控制是否使用基因查询优化方法的更精细的控制方法。只有当涉及的FROM关系数量至少有geqo_threshold个时,才使用基因查询优化。对于数量小于此值的查询,也许使用判定性的穷举搜索更有效。但是对于有许多表的查询,规划器做判断要花费很长时间。默认是“12”。请注意,一个FULL OUTER JOIN只算一个FROM项。
geqo_effort integer 控制GEQO中规划时间和查询规划的有效性之间的平衡。该变量必须是一个从1到10的整数。默认值是“5”。大的数值增加花费在进行查询规划上的时间,但是也很可能提高选中更有效的查询规划的概率。geqo_effort实际上并没有直接干什么事情;只是用于计算其他影响GEQO行为变量的缺省值(在下面描述)。如果需要,可以手动设置其他参数。
geqo_pool_size integer 控制GEQO使用的池大小。池大小是基因全体中的个体数量,它必须至少是“2”,有用的数值通常在100到1000之间。如果把它设置为“0”(默认值),那么就会基于geqo_effort和查询中表的数量选取一个合适的值。
geqo_generations integer 控制GEQO使用的子代数目。子代的意思是算法的迭代次数。它必须至少是“1”。有用值的范围和池大小相同。如果设置为“0”(默认),那么将基于geqo_pool_size选取合适的值。
geqo_selection_bias float 控制GEQO使用的选择性偏好。选择性偏好是在一个种群中的选择性压力。数值可以在1.5到2.0之间。(默认“2.0”)
geqo_seed float 控制GEQO使用的随机数产生器的初始值,用以选择随机路径。这个值可以在从0(默认)到1之间。修改此值会改变连接路径搜索的设置,同时可能会找到最优或最差路径。

当没有很多表做关连查询时, 并不需要关注这些基因查询优化器的参数, 因为此时基本不会走基因查询, 只有当关连查询表的数目超过“geqo_threshold”配置项时才会走基因查询优化算法。
如果不清楚基因查询的原理, 不能理解以上参数, 保留它们的默认值就可以了。

2.4 其他执行计划配置项

其他与执行计划相关的配置项如下表:

参数名称 类型 说明
default_statistics_target integer 此参数设置表字段的默认直方图统计目标值。如果表字段的直方图统计目标值未用ALTER TABLE SET STATISTICS明确设置,则使用此参数指定的值。此值越大,ANALYZE需要花费的时间越长,同时统计出的直方图信息也越详细,这样生成的执行计划也越准确。默认值是“100”,最大值是“10000”。
constraint_exclusion enum 指定在执行计划中是否使用约束排除。可以取3个值:“partition”、“on”、“off”。默认值为“partition”。约束排除就是指优化器分析WHERE子句中的过滤条件与表上的CHECK约束,当从语义上就能分析出而不需要访问这张表时,执行计划会直接跳过这张表。例如,表上的一个字段有约束“check col1 > 10000”,当查询“SELECT * FROM t WHERE col1 < 900”时,优化器对比约束条件知道根本没有符合“col1 < 900”条件的记录,因此会跳过对表的扫描直接返回0条记录。当优化器使用约束排除时,需要花费更多的时间去对比约束条件和WHERE子句中的过滤条件。在大多数情况下,对无继承的表打开约束排除意义不大,所以PostgreSQL把此值默认设置为“partition”。当对一张表做查询时,如果这张表有很多继承的子表,通常也需要扫描这些子表,设置为“partition”,优化器会对这些子表做约束排除分析。
cursor_tuple_fraction float 游标在选择执行计算时有两种策略:第一种是选择总体执行代价最小的;第二种是返回第一条记录时代价最小的。有时总体执行代价最小,但返回第一条记录的代价不是最小的,这时返回给用户的第一条记录的时间比较长,这会让用户觉得等待较长的时间系统才会响应,从而导致用户体验不太好。为了让用户体验比较好,可以选择返回第一条记录代价最小的执行计划,这时用户可以比较快地看到第一条记录。 设置游标,在选择总体代价最小的执行计划和返回第一条记录代价最小的执行计划两者之间比较倾向性的大小。默认值是“0.1”。最大值是“1.0”,此时游标会选择总体代价最小的执行计划,而不考虑多久才会输出第一个行。
from_collapse_limit integer 默认值是“8”。如果查询重写生成的FROM后的项目数不超过限制数目,优化器将把子查询融合到上层查询。小的数值缩短规划的时间,但是可能会生成差一些的查询计划。将此值设置得与配置项“geqo_threshold”的数值相同或更大,可能触发使用GEQO规划器,从而产生不确定的执行计划。
join_collapse_limit integer 如果查询重写生成的FROM后的项目数不超过限制数目,优化器把显式使用JOIN子句(不包括FULL JOIN)的连接也重写到FROM后的列表中。小的数值缩短规划的时间,但是可能会生成差一些的查询计划。默认值与from_collapse_limit相同。将此值设置得与配置项“geqo_threshold”的数值相同或更大,可能会触发使用GEQO规划器,从而产生不确定的执行计划。

3. 统计信息的收集

信息主要是AutoVacuum进程收集的, 用于查询优化时的代价估算。
表和索引的行数、 块数等统计信息记录在系统表“pg_class”中, 其他的统计信息主要收集在系统表“pg_statistic”中。
而Stats Collector子进程是PostgreSQL中专门的性能统计数据收集器进程, 其收集的性能数据可以通过“pg_stat_*”视图来查看, 这些性能统计数据对数据库活动的监控及分析性能有很大的帮助。

3.1 统计信息收集器的配置项

统计信息收集器的配置项如下表:

参数名称 类型 说明
track_counts boolean 控制是否收集表和索引上的访问的统计信息。默认是打开的。
track_functions enum 是否收集函数调用次数和时间的统计信息??梢匀 皀one”、“pl”、“all”3个值?!皀one”表示不收集,“pl”表示只收集过程语言函数,“all”表示收集所有的函数,包括SQL和C语言函数。默认值为“none”。
track_activities boolean 是否允许跟踪每个session正在执行的SQL命令的信息和命令开始执行的时间。这些信息可以在视图“pg_stat_activity”中看到。此参数默认是打开的。
track_activity_query_size integer 在pg_stat_activity视图中的query字段最多显示多少字节,默认值是“1024”,超过此设置的内容会被截断。
track_io_timing boolean 是否允许统计I/O调用的时间,默认为关掉。如果打开此选项,在带“BUFFERS”选项的EXPLAIN命令中将显示I/O调用的时间。这些I/O统计信息也可以在pg_stat_database和pg_stat_statements中看到,这是PostgreSQL 9.2版本之后新增加的参数。
update_process_title boolean 当后台服务进程正在执行命令(如一条SQL语句)时,是否更新其title信息。在Linux环境下此参数默认是打开的,所以在Linux环境下,默认可以使用ps命令查看一个后台服务进程是否正在执行命令。
stats_temp_directory string 设置存储临时统计数据的路径,可以是一个相对于数据目录的相对路径,也可以是一个绝对路径。默认值是“pg_stat_tmp”。

3.2 SQL执行的统计信息输出

可以使用以下4个boolean类型的参数来控制是否输出SQL执行过程的统计信息到日志中:

  • log_statement_stats
  • log_parser_stats
  • log_planner_stats
  • log_executor_stats

参数“log_statement_stats”控制是否输出所有SQL语句的统计信息, 其他的参数控制每个SQL命令是否输出不同执行模块中的统计信息。

3.3 手动收集统计信息

手动收集统计信息的命令是ANALYZE命令, 此命令用于收集表的统计信息, 然后把结果保存在系统表“pg_statistic”中。
优化器可以使用收集到的统计信息来确定最优的执行计划。

在默认的PostgreSQL配置中, AutoVacuum守护进程是打开的, 它能自动分析表、 收集表的统计信息。
当AutoVacuum进程关闭时, 需要周期性地, 或者在表的大部分内容变更后运行ANALYZE命令。
准确的统计信息能帮助优化器生成最优的执行计划, 从而改善查询的性能。

比较常用的一种策略是每天在数据库比较空闲的时候运行一次VACUUM和ANALYZE命令。

ANALYZE命令的语法格式如下:

ANALYZE [ VERBOSE ] [ table [ ( column [, ...] ) ] ]

命令中的选项说明如下。

  • VERBOSE: 增加此选项将显示处理的进度以及表的一些统计信息。
  • table: 要分析的表名, 如果不指定, 则对整个数据库中的所有表进行分析。
  • column: 要分析的特定字段的名称。 默认分析所有字段。

ANALYZE命令的应用示例如下。
只分析表“test01”中的“id2”列:

osdba=# ANALYZE test01(id2);
ANALYZE

分析表“test01”中的“id1”和“id2”两个列:

osdba=# ANALYZE test01(id1,id2);
ANALYZE

分析表“test01”中的所有列:

osdba=# ANALYZE test01;
ANALYZE

ANALYZE命令只需在表上加一个读锁, 因此它可以与表上的其他SQL命令并发执行。

ANALYZE命令会收集表的每个字段的直方图和最常用数值的列表。

对于大表, ANALYZE命令只读取表的部分内容做一个随机抽样, 不读取表的所有内容, 这样就保证了即使是在很大的表上也只需要很少时间就可以完成统计信息的收集。

统计信息只是近似的结果, 即使表内容实际上没有改变, 运行ANALYZE命令后EXPLAIN命令显示的执行计划中的COST值也会有一些变化。
为了增加所收集的统计信息的准确度,可以增大随机抽样比例, 这可以通过调整参数“default_statistics_target”来实现, 该参数可在session级别设置, 比如在分析不同的表时设置不同的值。

在下面的示例中, 假设表“test01”的行数较少, 设置“default_statistics_target”为“500”, 然后分析test01表, 表“test02”行数较多, 设置“default_statistics_target”为“10”, 再分析test02表, 命令如下:

osdba=# set default_statistics_target to 500;
SET

osdba=# analyze test01;
ANALYZE

osdba=# set default_statistics_target to 10;
SET

osdba=# analyze test02;
ANALYZE

也可以直接设置表的每个列的统计target值, 命令如下:

osdba=# ALTER TABLE test01 ALTER COLUMN id2 SET STATISTICS 200;
ALTER TABLE

ANALYZE命令的一个统计项是估计出现在每列的不同值的数目。
仅仅抽样部分行,该统计项的估计值有时会很不准确, 为了避免因此导致差的查询计划, 可以手动指定这个列有多少个唯一值, 其命令是“ALTER TABLE...ALTER COLUMN...SET (n_distinct=...)”,
示例如下:

osdba=# ALTER TABLE test01 ALTER COLUMN id2 SET (n_distinct=2000);
ALTER TABLE

另外, 如果表是有继承关系的其他子表的父表, 还可以设置“n_distinct_inherited”, 这样子表会继续父表的设置值, 示例如下:

osdba=# ALTER TABLE test01 ALTER COLUMN id2 SET (n_distinct_inherited=2000);
ALTER TABLE
最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容