`

PostgreSQL-学习-06--索引

阅读更多

PostgreSQL 提供了好几种索引类型:B-tree, Hash, GiST, GIN 。每种索引类型都比较适合某些特定的查询类型,因为它们用了不同的算法。

B-tree

CREATE TABLE test1 (

        id integer,

        content varchar

    );

    CREATE INDEX test1_id_index ON test1 (id);    

    B-Tree索引主要用于等于和范围查询,特别是当索引列包含操作符" <、<=、=、>=和>"作为查询条件时,PostgreSQL的查询规划器都会考虑使用B-Tree索引。在使用BETWEEN、IN、IS NULL和IS NOT NULL的查询中,PostgreSQL也可以使用B-Tree索引。然而对于基于模式匹配操作符的查询,如LIKE、ILIKE、~和 ~*,仅当模式存在一个常量,且该常量位于模式字符串的开头时,如col LIKE 'foo%'或col ~ '^foo',索引才会生效,否则将会执行全表扫描,如:col LIKE '%bar'。

Hash

CREATE INDEX name ON table USING hash (column);

    散列(Hash)索引只能处理简单的等于比较。当索引列使用等于操作符进行比较时,查询规划器会考虑使用散列索引。

    这里需要额外说明的是,PostgreSQL散列索引的性能不比B-Tree索引强,但是散列索引的尺寸和构造时间则更差。另外,由于散列索引操作目前没有记录WAL日志,因此一旦发生了数据库崩溃,我们将不得不用REINDEX重建散列索引

GiST

不是单独一种索引类型,而是一种架构,可以在这种架构上实现很多不同的索引策略。因此,可以使用 GiST 索引的特定操作符类型高度依赖于索引策略(操作符类)。

作为示例,PostgreSQL 的标准发布中包含用于二维几何数据类型的 GiST 操作符类,它支持<<、&<、&>、>>、<<|、&<|、|&>、|>>、@>、<@、~=、&& 操作符的索引查询。这些操作符的含义参见。 许多其它 GiST 操作符类位于 contrib 中,或者是单独的项目,更多信息参见。

GIN 

反转索引,它可以处理包含多个键的值(比如数组)。与 GiST 类似,GIN 支持用户定义的索引策略,可以使用 GIN 索引的特定操作符类型根据索引策略的不同而不同。、

作为示例,PostgreSQL 的标准发布中包含用于一维数组的 GIN 操作符类,它支持<@、@>、=、&& 操作符的索引查询。这些操作符的含义参见。 许多其它 GIN 操作符类位于 contrib, tsearch2, intarray 模块。更多信息参见

 

二、复合索引

PostgreSQL中的索引可以定义在数据表的多个字段上,如:

    CREATE TABLE test2 (

        major int,

        minor int,

        name varchar

    }

    CREATE INDEX test2_mm_idx ON test2 (major, minor);

    在当前的版本中,只有B-tree、GiST和GIN支持复合索引,其中最多可以声明32个字段。

    1. B-Tree类型的复合索引:

    在B-Tree类型的复合索引中,该索引字段的任意子集均可用于查询条件,不过,只有当复合索引中的第一个索引字段(最左边)被包含其中时,才可以获得最高效率。

    

    2. GiST类型的复合索引:

    在GiST类型的复合索引中,只有当第一个索引字段被包含在查询条件中时,才能决定该查询会扫描多少索引数据,而其他索引字段上的条件只是会限制索引返回的条目。假如第一个索引字段上的大多数数据都有相同的键值,那么此时应用GiST索引就会比较低效。 

 

    3. GIN类型的复合索引:

    与B-Tree和GiST索引不同的是,GIN复合索引不会受到查询条件中使用了哪些索引字段子集的影响,无论是哪种组合,都会得到相同的效率。

 

    使用复合索引应该谨慎。在大多数情况下,单一字段上的索引就已经足够了,并且还节约时间和空间。除非表的使用模式非常固定,否则超过三个字段的索引几乎没什么用处。 

 

三、组合多个索引:

 

    PostgreSQL可以在查询时组合多个索引(包括同一索引的多次使用),来处理单个索引扫描不能实现的场合。与此同时,系统还可以在多个索引扫描之间组成AND和OR的条件。比如,一个类似WHERE x = 42 OR x = 47 OR x = 53 OR x = 99的查询,可以被分解成四个独立的基于x字段索引的扫描,每个扫描使用一个查询子句,之后再将这些扫描结果OR在一起并生成最终的结果。另外一个例子是,如果我们在x和y上分别存在独立的索引,那么一个类似WHERE x = 5 AND y = 6的查询,就会分别基于这两个字段的索引进行扫描,之后再将各自扫描的结果进行AND操作并生成最终的结果行。

    为了组合多个索引,系统扫描每个需要的索引,然后在内存里组织一个BITMAP,它将给出索引扫描出的数据在数据表中的物理位置。然后,再根据查询的需要,把这些位图进行AND或者OR的操作并得出最终的BITMAP。最后,检索数据表并返回数据行。表的数据行是按照物理顺序进行访问的,因为这是位图的布局,这就意味着任何原来的索引的排序都将消失。如果查询中有ORDER BY子句,那么还将会有一个额外的排序步骤。因为这个原因,以及每个额外的索引扫描都会增加额外的时间,这样规划器有时候就会选择使用简单的索引扫描,即使有多个索引可用也会如此。     

 

    

四、唯一索引:

 

    目前,只有B-Tree索引可以被声明为唯一索引。

    CREATE UNIQUE INDEX name ON table (column [, ...]);

    如果索引声明为唯一索引,那么就不允许出现多个索引值相同的行。我们认为NULL值相互间不相等。

    

五、表达式索引:

 

    表达式索引主要用于在查询条件中存在基于某个字段的函数或表达式的结果与其他值进行比较的情况,如:

    SELECT * FROM test1 WHERE lower(col1) = 'value';

    此时,如果我们仅仅是在col1字段上建立索引,那么该查询在执行时一定不会使用该索引,而是直接进行全表扫描。如果该表的数据量较大,那么执行该查询也将会需要很长时间。解决该问题的办法非常简单,在test1表上建立基于col1字段的表达式索引,如:

    CREATE INDEX test1_lower_col1_idx ON test1 (lower(col1));

    如果我们把该索引声明为UNIQUE,那么它会禁止创建那种col1数值只是大小写有区别的数据行,以及col1数值完全相同的数据行。因此,在表达式上的索引可以用于强制那些无法定义为简单唯一约束的约束。现在让我们再看一个应用表达式索引的例子。

    SELECT * FROM people WHERE (first_name || ' ' || last_name) = 'John Smith';

    和上面的例子一样,尽管我们可能会为first_name和last_name分别创建独立索引,或者是基于这两个字段的复合索引,在执行该查询语句时,这些索引均不会被使用,该查询能够使用的索引只有我们下面创建的表达式索引。

    CREATE INDEX people_names ON people ((first_name || ' ' || last_name));

    CREATE INDEX命令的语法通常要求在索引表达式周围书写圆括弧,就像我们在第二个例子里显示的那样。如果表达式只是一个函数调用,那么可以省略,就像我们在第一个例子里显示的那样。

    从索引维护的角度来看,索引表达式要相对低效一些,因为在插入数据或者更新数据的时候,都必须为该行计算表达式的结果,并将该结果直接存储到索引里。然而在查询时,PostgreSQL就会把它们看做WHERE idxcol = 'constant',因此搜索的速度等效于基于简单索引的查询。通常而言,我们只是应该在检索速度比插入和更新速度更重要的场景下使用表达式索引。 

    

六、部分索引:

 

    部分索引(partial index)是建立在一个表的子集上的索引,而该子集是由一个条件表达式定义的(叫做部分索引的谓词)。该索引只包含表中那些满足这个谓词的行。

    由于不是在所有的情况下都需要更新索引,因此部分索引会提高数据插入和数据更新的效率。然而又因为部分索引比普通索引要小,因此可以更好的提高确实需要索引部分的查询效率。见以下三个示例:

    1. 索引字段和谓词条件字段一致:

    CREATE INDEX access_log_client_ip_ix ON access_log(client_ip)

        WHERE NOT (client_ip > inet '192.168.100.0' AND client_ip < inet '192.168.100.255');

    下面的查询将会用到该部分索引:

    SELECT * FROM access_log WHERE url = '/index.html' AND client_ip = inet '212.78.10.32';

    下面的查询将不会用该部分索引:

    一个不能使用这个索引的查询可以是∶

    SELECT * FROM access_log WHERE client_ip = inet '192.168.100.23';

 

    2. 索引字段和谓词条件字段不一致:

    PostgreSQL支持带任意谓词的部分索引,唯一的约束是谓词的字段也要来自于同样的数据表。注意,如果你希望你的查询语句能够用到部分索引,那么就要求该查询语句的条件部分必须和部分索引的谓词完全匹配。 准确说,只有在PostgreSQL能够识别出该查询的WHERE条件在数学上涵盖了该索引的谓词时,这个部分索引才能被用于该查询。

    CREATE INDEX orders_unbilled_index ON orders(order_nr) WHERE billed is not true;

    下面的查询一定会用到该部分索引:

    SELECT * FROM orders WHERE billed is not true AND order_nr < 10000;

    那么对于如下查询呢?

    SELECT * FROM orders WHERE billed is not true AND amount > 5000.00;

    这个查询将不像上面那个查询这么高效,毕竟查询的条件语句中没有用到索引字段,然而查询条件"billed is not true"却和部分索引的谓词完全匹配,因此PostgreSQL将扫描整个索引。这样只有在索引数据相对较少的情况下,该查询才能更有效一些。

    下面的查询将不会用到部分索引。

    SELECT * FROM orders WHERE order_nr = 3501;

    

    3. 数据表子集的唯一性约束:

    CREATE TABLE tests (

        subject text,

        target text,

        success boolean,

        ...

    );

    CREATE UNIQUE INDEX tests_success_constraint ON tests(subject, target) WHERE success;

    该部分索引将只会对success字段值为true的数据进行唯一性约束。在实际的应用中,如果成功的数据较少,而不成功的数据较多时,该实现方法将会非常高效。

     

七、检查索引的使用:

 

    见以下四条建议:

    1. 总是先运行ANALYZE。

    该命令将会收集表中数值分布状况的统计。在估算一个查询返回的行数时需要这个信息,而规划器则需要这个行数以便给每个可能的查询规划赋予真实的开销值。如果缺乏任何真实的统计信息,那么就会使用一些缺省数值,这样肯定是不准确的。因此,如果还没有运行ANALYZE就检查一个索引的使用状况,那将会是一次失败的检查。 

    2. 使用真实的数据做实验。

    用测试数据填充数据表,那么该表的索引将只会基于测试数据来评估该如何使用索引,而不是对所有的数据都如此使用。比如从100000行中选1000行,规划器可能会考虑使用索引,那么如果从100行中选1行就很难说也会使用索引了。因为100行的数据很可能是存储在一个磁盘页面中,然而没有任何查询规划能比通过顺序访问一个磁盘页面更加高效了。与此同时,在模拟测试数据时也要注意,如果这些数据是非常相似的数据、完全随机的数据,或按照排序顺序插入的数据,都会令统计信息偏离实际数据应该具有的特征。    

    3. 如果索引没有得到使用,那么在测试中强制它的使用也许会有些价值。有一些运行时参数可以关闭各种各样的查询规划。

    4. 强制使用索引用法将会导致两种可能:一是系统选择是正确的,使用索引实际上并不合适,二是查询计划的开销计算并不能反映现实情况。这样你就应该对使用和不使用索引的查询进行计时,这个时候EXPLAIN ANALYZE命令就很有用了 

分享到:
评论

相关推荐

    postgresql-9.6.11-1-windows-x64

    postgresql-9.6.11-1-windows-x64位可安装,用于学习交流,下载安装可用 请勿用于商业用途,如有版权纠纷,本人概不负责! 1. PostgreSQL完全免费,而且是BSD协议. 2. 与PostgreSQl配合的开源软件很多,有很多分布式...

    postgresql 中文学习手册

    PostgreSQL学习手册(索引) PostgreSQL学习手册(事物隔离) PostgreSQL学习手册(性能提升技巧原理) PostgreSQL学习手册(服务器配置) PostgreSQL学习手册(角色和权限) PostgreSQL学习手册(数据库管理) PostgreSQL学习...

    PostgreSQL中文手册9.2

    PostgreSQL PostgreSQL PostgreSQL学习手册 学习手册 学习手册 (索引 ) 42 一、索引的类型: 一、索引的类型: 一、索引的类型: 一、索引的类型: .42 四、唯一索引: 四、唯一索引: 四、唯一索引: .43 ...

    Postgis-学习 入门 教程 教学 手册-Postgresql

    PostGIS简介、安装、新建空间数据库、简单SQL语句查询练习、几何数据操作和练习、空间关系操作与练习、空间链接与练习、投影数据与练习、空间索引与练习…… 还包括PostgreSQL for spatial的配置,验证,postgis函数...

    Mastering PostgreSQL 15-2023

    详细讲述了PostgreSQL 15版本的事务和锁、索引、高级SQL、日志和系统统计信息、优化查询以获得良好性能、编写存储过程、安全、备份与恢复、备份与复制、决定有用的扩展、疑难解答、数据库迁移……适合中级以上编程、...

    Learn-PostgreSQL:学习PostgreSQL,由Packt发布

    本书涵盖以下激动人心的功能: 了解如何通过运行PostgreSQL实例来管理用户和连接使用服务器端编程与事务边界进行交互识别瓶颈以有效维护数据库创建和管理扩展以向集群添加新功能为每种情况选择最佳的索引类型如果您...

    crack-detection:裂缝检测系统可检测道路裂缝的百分比并警告您车辆的速度限制

    裂纹检测系统 该项目使用 Python 的 OpenCV 库和深度学习来检测裂纹区域,并根据裂纹百分比警告驾驶员所需...PostgreSQL - 数据库 安装和设置步骤 克隆存储库 git clone https://github.com/ak2502/crack-detection.git

    learn-psql:包含PostgreSQL命令的存储库,可用于开发和学习

    还支持部分索引和位图索引。 提供多种设置触发器的选项。 :thinking_face: PostgreSQL在哪里使用? LAPP堆栈中的健壮数据库。 它代表Linux,Apache,PostgreSQL和PHP(或Python和Perl)。 PostgreSQL主要用作强大的...

    SQLServer迁移PG经验董红禹.pdf

    云栖TechDay - PostgreSQL天天象上活动-上海站演讲者:董红禹 主要章节 背景 迁移工作流程 数据类型转换 函数转换 临时表 视图、外键、索引 partial index 部分索引 非索引列的使用 表达式索引 应用SQL上修改 ...

    matlab小动物图形代码-Awesome-GeoAI:真棒

    Ag领域中有关机器学习/深度学习的资源,帖子,项目,数据集 内容 平台类 -- -- 纸本密码 | Github(2016)-MODIS预测全球未来的植被健康价值 帖子 :glowing_star: -PostgreSql空间扩展。 -一组PostgreSQL函数,在...

    Mastering PostgreSQL 13

    本书详细讲述了数据库管理管理增强(EXPLAIN、Auto_explain、autovacuum、pg_stat_statements),新特性逻辑复制支持分区表、持续的性能提升Btree索引优化(引入Deduplication技术)、支持增量排序(Incremental ...

    Greenplum数据库架构.pdf

    学习 · 创造 · 超越 · 共享 Greenplum 架构 P9平台组 林舒杨 PostgreSQL数据库 基于BSD许可证发布 1985年开始于UC Berkeley的Ingres计 划 1995.7.21发布0.03版本 2014.3.20发布9.3.4版本 PostgreSQL特征 函数: ...

    django-docs-2.2-zh-hans.zip

    其它: 支持的数据库 | 旧数据库 | 提供初始化数据 | 优化数据库访问 | PostgreSQL 的特定功能 视图层 Django 具有 “视图” 的概念,负责处理用户的请求并返回响应。通过以下链接查找所有你需要知道的有关视图的...

    知识:my学习主题索引

    知识:my学习主题索引

    oracle学习文档 笔记 全面 深刻 详细 通俗易懂 doc word格式 清晰 连接字符串

    PostgreSQL 号称“世界上最先进的开源数据库“,可以运行在多种平台下,是tb级数据库,而且性能也很好 中大型企业 oracle 甲骨文 获得最高认证级别的ISO标准安全认证,性能最高, 保持开放平台下的TPC-D和TPC-C的...

    PostgreSQL教程(十四):数据库维护

    主要介绍了PostgreSQL教程(十四):数据库维护,本文讲解了恢复磁盘空间、更新规划器统计、VACUUM和ANALYZE的示例、定期重建索引等内容,需要的朋友可以参考下

    数据库实战的教程.doc

    4. 学习数据库优化:了解数据库优化的方法,包括索引的使用、数据库的备份和恢复等。 5. 实战演练:通过实际操作进行练习,可以参考一些开源的数据库管理系统,如MySQL、PostgreSQL等,进行实际操作。 具体的教程...

    navicat html help

    Navicat 的功能足以符合专业开发人员的所有需求,但是对数据库服务器的新手来说又相当容易学习。有了极完备的图形用户介面 (GUI),Navicat 让你可以简便地以安全且简单的方法创建、组织、访问和共享信息。 Navicat ...

    java版飞机大战源码-free-programming-books-zh_CN:复制学习

    java版飞机大战源码 ...PostgreSQL MySQL 管理和监控 项目相关 (译言网) 设计模式 Web 大数据 编程艺术 【第一部分】 (豆瓣阅读,免费书籍) 其它 Android Google Material Design 正體中文版 ( ) Android 一些重要知

    SQL必知必会(第3版)--详细书签版

    本书涉及不同平台上数据的排序、过滤和分组,以及表、视图、联结、子查询、游标、存储过程和触发器等内容,通过本书读者可以系统地学习到sql 的知识和方法。  本书注重实用性,操作性很强,适合于sql 的初学者学习...

Global site tag (gtag.js) - Google Analytics