暂无图片
暂无图片
6
暂无图片
暂无图片
暂无图片

[译] 通过Postgres索引创建理解操作符类及索引类型等概念

原创 多米爸比 2022-02-15
2056

原文地址:https://pganalyze.com/blog/postgres-create-index
原文作者:Lukas Fittl
翻译:多米爸比

翻译说明:非完全按照原文进行全部翻译,且原文有一处错误,细心读者可以对比查找。

前言

大多数开发人员面临这样的挑战:他们刚编写的程序部署到生产环境后,应用系统突然变慢了。经过一些工具及监控系统的分析,发现是新编的代码里使用新的Query语句引起。进一步分析根因是由Query不能使用index引起。

那什么时候Query可以使用索引,再Postgres里我们如何正确的创建索引呢。本文我们将通过CREATE
INDEX命令详细了解它各个有用的方面,同时我们也会分析PostgreSQL查询的操作符及数据类型,这样我们就可以如何选择最佳的索引定义。

image.png
本文将主要围绕这张图展开

我们如何创建索引

首先,我们来看一下Postgres里面创建索引的基本方式,它的语法是

CREATE INDEX ON [table] ([column1]); 

示例:比如用户表有一个针对email地址的查询语句。

SELECT * FROM users WHERE users.email = 'test@example.com';

我们知道这个语句是针对email这一列进行查询,因此我们的索引应该创建在用户表的这个特定列上。

CREATE INDEX ON users (email);

当我们执行create index on users命令之后,Postgres就帮我们创建好了索引。

需要注意的是:索引只是一个冗余的数据结构,也就是说我们即使删除了刚才创建的索引,我们也不会丢失任何的数据。使用索引的好处是帮助我们加速搜索表上特定的一些记录行。

Postgres如何领会Query

当Postgres运行用户的语句时,会经历多个阶段,大概可以分成下面四个阶段:

  1. Parsing
  2. Parse analysis
  3. Planning
  4. Execution

在上面这些阶段里,Query语句不再仅仅只是一个文本串,而是一个语法tree,每个阶段都会修改这棵树以及对它的结构进行注解,直到最后的执行阶段。

为了理解Postgres数据库是如何使用索引,我们首先需要理解Parse analysis阶段做了哪些事情。

我们再看稍微复杂一点的示例:

SELECT * FROM users WHERE users.email = 'test@example.com' AND users.deleted_at IS NULL;

根据email地址的查询语句,同时多了一个查询条件。

我们如果打开数据库debug_print_parse参数(生产环境禁用),可以从数据库日志看到详细的日志内容。

LOG:  parse tree:
DETAIL:     {QUERY 
	   ...
	      :quals 
	         {BOOLEXPR 
	         :boolop and 
	         :args (
	            {OPEXPR 
	            :opno 98 
	            :opfuncid 67 
	            :opresulttype 16 
	            :opretset false 
	            :opcollid 0 
	            :inputcollid 100 
	            :args (
	               ...
	            )
	            :location 38
	            }
	            {NULLTEST 
	            :arg 
	               ...
	            :nulltesttype 0 
	            :argisrow false 
	            :location 80
	            }
...

日志输出的格式很难直接读懂,但我们可以通过下面的可视化来展示。
image.png

我们可以看到两个重要的解析节点,WHERE子句里的OpExpr节点和NullTest节点。首先来关注第一个OpExpr节点。

理解操作符Operators和数据类型Data Types

数据类型我们一般都很熟悉,这个我们再schema里面创建表时都用到过。操作符Operator定义了一个数据类型值或者多个数据类型值如何进行比较。OpExpr节点就代表着使用一个操作符去比较一个或多个给定的数据类型值的表达式。

在示例中我们使用的操作符可以从pg_operator系统表查询。

SELECT oid, oid::regoperator, oprcode, oprnegate::regoperator
  FROM pg_operator
 WHERE oprname = '=' AND oprleft = 'text'::regtype AND oprright = 'text'::regtype;
oid |     oid      | oprcode |   oprnegate   
-----+--------------+---------+---------------
 98 | =(text,text) | texteq  | <>(text,text)
(1 row)

从上面也可以看到操作符对应的内部实现函数。如果我们想弄清楚它的实现函数,可以查看texteq函数的源码。

/*
 * Comparison functions for text strings.
 */
Datum
texteq(PG_FUNCTION_ARGS)
{
    ...
    if (lc_collate_is_c(collid) ||
		collid == DEFAULT_COLLATION_OID ||
		pg_newlocale_from_collation(collid)->deterministic)
    {
        ...
    	result = (memcmp(VARDATA_ANY(targ1), VARDATA_ANY(targ2),
				  len1 - VARHDRSZ) == 0);
        ...
    }
    else
	{
        ...
        result = (text_cmp(arg1, arg2, collid) == 0);
        ...
    }
    ...
}

当然我们也可以创建自定义的操作符来为自定义数据类型服务。

操作符是创建正确索引最基本且最重要的因素之一。它决定我们如何搜索数据表的值。例如我们可以使用等号操作符来匹配输入值是否与列值相等,也可以使用两个@符号来对全文搜索列执行一个文本text值的匹配。

寻找正确的索引类型

那当我们使用不同的索引类型时,它的数据结构也限定了它所支持的操作符。比如Postgres里最常用的索引类型是B-tree索引,它支持等号、范围比较<, <=, =>, >, ~ , ~*。

我们再来看一个例子,users表的全文搜索列上,我们需要使用两个@操作符来搜索这个列。

SELECT * FROM users WHERE about_text_search @@ to_tsquery('index');

如果我们只是简单的对这个列创建索引,不指定索引类型,那我们的查询语句还是会跟没创建索引一样继续使用全表扫描。

CREATE INDEX ON users(about_text_search);
pgaweb=# EXPLAIN SELECT * FROM users WHERE about_text_search @@ to_tsquery('index');
                                 QUERY PLAN                                 
----------------------------------------------------------------------------
 Seq Scan on users  (cost=10000000000.00..10000000006.51 rows=1 width=4463)
   Filter: (about_text_search @@ to_tsquery('index'::text))
(2 rows)

因为通用的b-tree索引并没有支持全文检索类型数据结构,也并没有支持该操作符的操作符类。我们可以反过来通过全文检索操作符来确定我们可以使用哪种索引类型。

SELECT am.amname AS index_method,
       opf.opfname AS opfamily_name,
       amop.amopopr::regoperator AS opfamily_operator
  FROM pg_am am,
       pg_opfamily opf,
       pg_amop amop
 WHERE opf.opfmethod = am.oid AND amop.amopfamily = opf.oid
       AND amop.amopopr = '@@(tsvector,tsquery)'::regoperator;
 index_method | opfamily_name |  opfamily_operator   
--------------+---------------+----------------------
 gist         | tsvector_ops  | @@(tsvector,tsquery)
 gin          | tsvector_ops  | @@(tsvector,tsquery)
(2 rows)

从上面的结果我们知道可以使用gist或者gin索引。那我们使用using gin创建索引之后,再次执行语句,可以看到能使用索引。

=# EXPLAIN SELECT * FROM users WHERE about_text_search @@ to_tsquery('index');
                                        QUERY PLAN                                         
-------------------------------------------------------------------------------------------
 Bitmap Heap Scan on users  (cost=8.25..12.51 rows=1 width=4463)
   Recheck Cond: (about_text_search @@ to_tsquery('index'::text))
   ->  Bitmap Index Scan on users_about_text_search_idx1  (cost=0.00..8.25 rows=1 width=0)
         Index Cond: (about_text_search @@ to_tsquery('index'::text))
(4 rows)

前面查询Postgres内部表时,还看到有一个tsvector_ops的名称,它有什么作用呢?

对于一个确定的操作符,可能会有多种不同的操作符分类,不同的操作符分类是指用于操作特定数据类型索引的最小运算符集,每个操作符分类定义着对某个指定索引类型的数据展现。

明确指定Operator Classes

image.png
我们来查询前面查询语句里用到的“=(text,text)”操作符

SELECT am.amname AS index_method,
       opf.opfname AS opfamily_name,
       amop.amopopr::regoperator AS opfamily_operator
  FROM pg_am am,
       pg_opfamily opf,
       pg_amop amop
 WHERE opf.opfmethod = am.oid AND amop.amopfamily = opf.oid
       AND amop.amopopr = '=(text,text)'::regoperator;
 index_method |  opfamily_name   | opfamily_operator 
--------------+------------------+-------------------
 btree        | text_ops         | =(text,text)
 hash         | text_ops         | =(text,text)
 btree        | text_pattern_ops | =(text,text)
 hash         | text_pattern_ops | =(text,text)
 spgist       | text_ops         | =(text,text)
 brin         | text_minmax_ops  | =(text,text)
 gist         | gist_text_ops    | =(text,text)
(7 rows)

创建索引时如果我们不显示指定operator class,也可以看到索引对应的默认operator class。文本类型的operator class是我们通常所使用的。再一些场景下我们也需要设置特定的operator class。

当我们数据库里运行一个like查询。DB上的collation不是大C,索引是不支持like操作的。

CREATE INDEX ON users (email);
pgaweb=# EXPLAIN SELECT * FROM users WHERE email LIKE 'lukas@%';
                                 QUERY PLAN                                 
----------------------------------------------------------------------------
 Seq Scan on users  (cost=10000000000.00..10000000001.26 rows=1 width=4463)
   Filter: ((email)::text ~~ 'lukas@%'::text)
(2 rows)

我们可以看到查询还是走全表扫描。如果我们进行引导,索引也能对这种场景生效:1)我们在数据库创建时指定collation为大C(也就是告诉数据库我们不想绑定具体语言的排序比较规则);2)使用文本模式操作符text_pattern_ops。

CREATE INDEX ON users (email text_pattern_ops);
pgaweb=# EXPLAIN SELECT * FROM users WHERE email LIKE 'lukas@%';
                                         QUERY PLAN                                         
--------------------------------------------------------------------------------------------
 Index Scan using users_email_idx on users  (cost=0.14..8.16 rows=1 width=4463)
   Index Cond: (((email)::text ~>=~ 'lukas@'::text) AND ((email)::text ~<~ 'lukasA'::text))
   Filter: ((email)::text ~~ 'lukas@%'::text)
(3 rows)

可以看到明确指定文本模式的操作符后,也能使用索引。

到这里我们知道了对列使用不同的索引类型以及操作符分类。接下来我们将讨论创建索引的一些其他方面。

使用多列属性

再创建索引的语句定义里面,还有一个基本的特性是使用多列选项。

CREATE INDEX ON [table] ([column_a], [column_b]);

对于多列属性,不同索引类型的数据结构有不同的展现形式,有些索引类型比如Hash索引、spgist索引是不支持多列属性的。

对于B-tree多列索引,列的顺序非常重要,如果部分查询语句只使用column_a,而大部分查询语句都使用column_b,那我们应该把column_b放在前面。如果我们不遵守这个规则,我们最终的查询需要跳过所有前导列。另一方面,如果我们使用GIST索引,那这个顺序就无所谓了,我们可以以任意顺序指定。

还有一个值得讨论的问题:我们是创建单个多列索引合适还是创建多个单列索引更好?

CREATE INDEX ON [table] ([column_a]);
CREATE INDEX ON [table] ([column_b]);
--- or
CREATE INDEX ON [table] ([column_a], [column_b]);

如果从单个查询语句的角度来看,大部分的答案是创建单个多列索引匹配度更高,单个多列索引也会比多个索引更快。但是如果我们是一个很大的工作负载,我们应该需要创建多个单列索引,同时我们也要意识到Postgres在这种场景下也会做更多的工作,我们也应该通过explain命令或者来实际验证索引的选择情况。

使用函数及表达式索引

除了对指定列创建索引,我们也可以对列的表达式创建索引,比较典型的例子是email嵌套小写函数之后的比较。

SELECT * FROM users WHERE lower(email) = $1

如果我们简单的对email列创建索引,通过explain去观察,会发现并不会走索引,因为它并不匹配表达式。

我们知道lower函数是一个“immutable”属性的函数,所以我们可以创建函数表达式索引。

CREATE INDEX ON users (lower(email));

使用表达式索引之后,我们的查询就可以使用索引了。注意并不是所有的函数都可用于创建表达式索引,比如使用now函数创建就会报错。

CREATE INDEX ON users (now());
ERROR:  functions in index expression must be marked IMMUTABLE

另外需要注意创建了表达式索引后,只有符合表达式的语句才能使用索引,如果只是引用独立的列,是不会走索引的。

运用WHERE子句创建部分索引

我们再回到刚开始的介绍,我们介绍了两个节点,第一个OpExpr节点已经介绍过了。我们再来看一下NullTest表达式。
image.png
我们的应用查询只需要返回没有被标记为删除的数据。根据我们的工作负载,有可能需要被忽略的记录(也就是打了删除标记的记录)非常多时,我们可以在创建索引时包含删除标记这一列,我们实际不需要关注有删除标记的数据。因为如果包含在索引条目里,那将是非常大的空间浪费。

Postgres针对这种场景有更好的解决方式:使用部分索引,我们可以限制索引条目的数据行。当然如果没有这个限制,数据行不会保存到索引中,这会节省空间。在查询执行期间,在很多案例里部分索引可以节省大量时间花销的,因为计划生成器会做简单的检测,判定部分索引是否匹配,忽略不匹配的所有数据。

实际生产中,我们需要做的是在索引的定义里增加一个where子句。

CREATE INDEX ON users(email) WHERE deleted_at IS NULL;

需要注意的是:面临下面两个问题时我们应该放弃使用部分索引。
第一:添加了部分索引的限制,意味着只有包含表达式的语句才可以使用索引。那如果我们既需要查询带限制条件,也需要不带限制条件,那我们可能要创建两个索引。
第二:添加大量的部分索引也会加重Postgres计划生成器的花销,因为它需要做大量昂贵的分析来判定应该使用哪个索引。

使用INCLUDE关键字创建Covering Index

下面我们介绍Postgres索引的另外一个新特性,使用include关键字来创建覆盖索引,让更多场景可以使用index-only扫描。

首先我们来看下index扫描和index-only扫描的区别。index-only扫描可以直接从索引中获取数据,而不需要再从磁盘中去读取数据。

另外index-only只有当表最近做过vacuum操作才会起作用。不然Postgres需要频繁的对每个索引条目去检查可见性,这样大多情况下Postgres会优先选择index scan。

让我们来看下面两个例子,上边的语句能使用index-only scan,下边的语句不能,因为查询的列不一样。

CREATE INDEX ON users (email, id);
=# EXPLAIN SELECT id FROM users WHERE email = 'test@example.com';
                                     QUERY PLAN                                      
-------------------------------------------------------------------------------------
 Index Only Scan using users_email_id_idx on users  (cost=0.14..4.16 rows=1 width=4)
   Index Cond: (email = 'test@example.com'::text)
(2 rows)
=# EXPLAIN SELECT id, fullname FROM users WHERE email = 'test@example.com';
                                    QUERY PLAN                                    
----------------------------------------------------------------------------------
 Index Scan using users_email_id_idx on users  (cost=0.14..8.15 rows=1 width=520)
   Index Cond: ((email)::text = 'test@example.com'::text)
(2 rows)

如果想让下边的语句也能使用index-only scan,在我们创建多列索引时,可以把fullname列添加进去。

CREATE INDEX ON users (email, id, fullname);
=# EXPLAIN SELECT id, fullname FROM users WHERE email = 'test@example.com';
                                           QUERY PLAN                                           
------------------------------------------------------------------------------------------------
 Index Only Scan using users_email_id_fullname_idx on users  (cost=0.14..4.16 rows=1 width=520)
   Index Cond: (email = 'test@example.com'::text)
(2 rows)

然而这样做却有一些小小的限制:比如我们要使用唯一索引、索引可能会发生膨胀。

此时使用include关键字会是一个更好的选择:

CREATE INDEX ON users (email, id) INCLUDE (fullname);

这会让我们的索引更加轻量,也不用担心唯一约束的问题,同时include添加的列也很明确,就是为了支持index only scan。

这个特性也需要非常节俭的去使用:因为随着我们添加更多的数据到索引中,索引的值也会非常的大,这也会产生问题,添加太多的列到include子句中其实并不是一个很好的方法。

生产环境安全地添加及删除索引

生产环境下,如果我们不使用CONCURRENTLY参数,则会获取exclusive lock,会阻塞表的读写操作。Postgres能够在不获取任何锁的情况下添加索引。它使用CONCURRENTLY参数来执行此操作。

CREATE INDEX CONCURRENTLY ON users (email) WHERE deleted_at IS NULL;

删除索引操作也是使用同样的关键字。

小结

通过本文我们应该基本了解与索引相关的操作符和操作符类,了解这些概念对于复杂查询创建最佳索引至关重要。我们还研究了CREATE INDEX命令的一些补充功能特性,这些特性为Postgre判定使用哪个索引非常关键。

本文实际上还有一些方面没有讨论:比如使用特定表空间创建索引,使用索引存储参数(尤其是对GIN索引类型特别有用!)以及指定列的排序顺序。作者也非常鼓励大家通过Postgres文档进一步阅读本文的一些概念。

保持联系

从2019年12月开始写第一篇文章,分享的初心一直在坚持,本人现在组建了一个PG乐知乐享交流群,欢迎关注我文章的小伙伴进群吹牛唠嗑,交流技术,互赞文章。

123.png

如果群二维码失效可以加我微信。

456.png

最后修改时间:2022-10-23 10:01:13
「喜欢这篇文章,您的关注和赞赏是给作者最好的鼓励」
关注作者
【版权声明】本文为墨天轮用户原创内容,转载时必须标注文章的来源(墨天轮),文章链接,文章作者等基本信息,否则作者和墨天轮有权追究责任。如果您发现墨天轮中有涉嫌抄袭或者侵权的内容,欢迎发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

评论