PostgreSQL建立索引如何避免写数据锁定

数据库 其他数据库 PostgreSQL
写这篇blog源自一个帅哥在建索引发生了表锁的问题。正常情况下Postgresql建立普通btree索引时会阻塞DML(insert,update,delete)操作,直到索引完成,期间读操作不受阻塞。

写这篇blog源自一个帅哥在建索引发生了表锁的问题。先介绍一下Postgresql的建索引语法:

Version:9.1

  1. CREATE [ UNIQUE ] INDEX [ CONCURRENTLY ] [ name ] ON table [ USING method ]  
  2.     ( { column | ( expression ) } [ COLLATE collation ] [ opclass ] [ ASC | DESC ] [ NULLS { FIRST | LAST } ] [, ...] )  
  3.     [ WITH ( storage_parameter = value [, ... ] ) ]  
  4.     [ TABLESPACE tablespace ]  
  5.     [ WHERE predicate ] 

这里不解释语法的诸多参数使用(排序,使用方法,填充因子等),主要说一下concurrently的使用场景。

正常情况下Postgresql建立普通btree索引时会阻塞DML(insert,update,delete)操作,直到索引完成,期间读操作不受阻塞。当只有一个用户操作这当然没问题,但是在生产环境,并发比较高的情况下,特别是大表建索引就不能这么操作了,不然用户要跳起来骂娘了,点个按钮一天还没反应过来。

--使用

Postgresql提供了一个参数,可以在线建立索引的时候避免因写数据而锁表,这个参数叫concurrently。使用很简单,就是用create index concurrently来代替create index即可。

--副作用

当然了,使用这个参数是有副作用的,不使用这个参数建索引时DB只扫描一次表,使用这个参数时,会引发DB扫两次表,同时等待所有潜在会读到该索引的事务结束,这么一来,系统的CPU和IO,内存等会受一点影响,所以综合考虑,仍然让用户自行选择,而不是默认。

--失败

在使用concurrently参数建索引时,有可能会遇到失败的情况,比如建唯一索引索引发现数据有重复,又或者用户发现建索引时建错字段的,取消建索引操作了。此时该表上会存在一个索引,这是因为带这个参数的建索引命令一经发出,就首先会在系统的日志表里先插一个索引记录进去,又因为这个索引最终建失败了,所以会被标记一个INVALID的状态,如下:

  1. postgres=# \d t_kenyon  
  2.        Table "public.t_kenyon" 
  3.  Column |  Type   | Modifiers   
  4. --------+---------+-----------  
  5.  col    | integer |   
  6. Indexes:  
  7.     "idx" btree (col) INVALID 

--重建

遇到上述失效的索引重建时两个办法,一个是drop index index_name,然后再执行create index concurrently。还有一个是执行reindex index_name命令,但是后者不支持concurrent参数。

--总结

在生产上执行创建索引命令时最好带上此参数,因为多消耗一点系统资源和时间来换取用户的不间断访问更新是相对值得的。 如果是索引重建,可以再在原基础上建立一个不同名的相同索引,然后取消老的索引。

英文原文: http://www.postgresql.org/docs/9.1/static/sql-createindex.html

译文链接:http://my.oschina.net/Kenyon/blog/93465

责任编辑:林师授 来源: OsChina
相关推荐

2015-09-14 10:03:23

云锁定网络服务云供应商

2021-05-21 10:50:39

云计算供应商锁定云迁移

2024-03-01 09:57:19

数据库检索项目

2022-10-12 13:33:25

PostgreSQL数据库

2014-10-15 10:01:12

2013-07-16 09:38:01

厂商锁定 云厂商锁定开放APISaaS厂商

2013-04-11 09:21:46

PaaS平台即服务PaaS产品

2018-04-17 12:23:37

多云云计算供应商

2021-04-22 11:22:12

云计算数据迁移混合云

2011-09-05 10:01:41

虚拟化

2011-05-18 13:16:21

MySQL数据库锁定

2017-10-24 14:48:05

数据数据驱动文化大数据

2023-09-14 23:05:57

​ ​MySQL数据库

2011-04-11 16:50:13

Oracle数据库索引

2012-05-07 15:32:46

Erlang

2010-11-29 10:22:46

Sybase建立索引

2023-03-27 09:57:00

PostgreSQL并发索引

2010-05-24 14:57:03

MySQL数据库表

2009-02-02 16:50:34

数据库表的锁定MySQL

2019-10-12 09:58:53

云计算IT服务云计算提供商
点赞
收藏

51CTO技术栈公众号