ИзползванС Π½Π° всички Ρ„ΡƒΠ½ΠΊΡ†ΠΈΠΈ Π½Π° индСкситС Π² PostgreSQL

ИзползванС Π½Π° всички Ρ„ΡƒΠ½ΠΊΡ†ΠΈΠΈ Π½Π° индСкситС Π² PostgreSQL
Π’ свСта Π½Π° Postgres индСкситС са ΠΎΡ‚ ΡΡŠΡ‰Π΅ΡΡ‚Π²Π΅Π½ΠΎ Π·Π½Π°Ρ‡Π΅Π½ΠΈΠ΅ Π·Π° Π΅Ρ„Π΅ΠΊΡ‚ΠΈΠ²Π½Π°Ρ‚Π° навигация Π² Ρ…Ρ€Π°Π½ΠΈΠ»ΠΈΡ‰Π΅Ρ‚ΠΎ Π½Π° Π±Π°Π·Π°Ρ‚Π° Π΄Π°Π½Π½ΠΈ (Π½Π°Ρ€Π΅Ρ‡Π΅Π½ΠΎ β€žΡ…ΠΈΠΉΠΏβ€œ). Postgres Π½Π΅ ΠΏΠΎΠ΄Π΄ΡŠΡ€ΠΆΠ° Π³Ρ€ΡƒΠΏΠΈΡ€Π°Π½Π΅ Π·Π° Π½Π΅Π³ΠΎ ΠΈ MVCC Π°Ρ€Ρ…ΠΈΡ‚Π΅ΠΊΡ‚ΡƒΡ€Π°Ρ‚Π° Π²ΠΈ ΠΊΠ°Ρ€Π° Π΄Π° сС ΠΎΠΊΠ°ΠΆΠ΅Ρ‚Π΅ с ΠΌΠ½ΠΎΠ³ΠΎ вСрсии Π½Π° Π΅Π΄ΠΈΠ½ ΠΈ ΡΡŠΡ‰ ΠΊΠΎΡ€Ρ‚Π΅ΠΆ. Π‘Π»Π΅Π΄ΠΎΠ²Π°Ρ‚Π΅Π»Π½ΠΎ Π΅ ΠΌΠ½ΠΎΠ³ΠΎ Π²Π°ΠΆΠ½ΠΎ Π΄Π° ΠΌΠΎΠΆΠ΅Ρ‚Π΅ Π΄Π° ΡΡŠΠ·Π΄Π°Π²Π°Ρ‚Π΅ ΠΈ ΠΏΠΎΠ΄Π΄ΡŠΡ€ΠΆΠ°Ρ‚Π΅ Π΅Ρ„Π΅ΠΊΡ‚ΠΈΠ²Π½ΠΈ индСкси Π·Π° ΠΏΠΎΠ΄Π΄Ρ€ΡŠΠΆΠΊΠ° Π½Π° прилоТСния.

Π•Ρ‚ΠΎ няколко ΡΡŠΠ²Π΅Ρ‚Π° Π·Π° ΠΎΠΏΡ‚ΠΈΠΌΠΈΠ·ΠΈΡ€Π°Π½Π΅ ΠΈ подобряванС Π½Π° ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Π½Π΅Ρ‚ΠΎ Π½Π° индСкси.

Π—Π°Π±Π΅Π»Π΅ΠΆΠΊΠ°: ΠΏΠΎΠΊΠ°Π·Π°Π½ΠΈΡ‚Π΅ ΠΏΠΎ-Π΄ΠΎΠ»Ρƒ заявки работят Π²ΡŠΡ€Ρ…Ρƒ Π½Π΅ΠΏΡ€ΠΎΠΌΠ΅Π½Π΅Π½ ΠΏΡ€ΠΈΠΌΠ΅Ρ€Π½Π° Π±Π°Π·Π° Π΄Π°Π½Π½ΠΈ Π½Π° страница.

ИзползванС Π½Π° ΠΏΠΎΠΊΡ€ΠΈΠ²Π°Ρ‰ΠΈ индСкси

НСка Ρ€Π°Π·Π³Π»Π΅Π΄Π°ΠΌΠ΅ заявка Π·Π° ΠΈΠ·Π²Π»ΠΈΡ‡Π°Π½Π΅ Π½Π° ΠΈΠΌΠ΅ΠΉΠ» адрСси Π·Π° Π½Π΅Π°ΠΊΡ‚ΠΈΠ²Π½ΠΈ ΠΏΠΎΡ‚Ρ€Π΅Π±ΠΈΡ‚Π΅Π»ΠΈ. Π’Π°Π±Π»ΠΈΡ†Π° customer ΠΈΠΌΠ° ΠΊΠΎΠ»ΠΎΠ½Π° active, Π° заявката Π΅ проста:

pagila=# EXPLAIN SELECT email FROM customer WHERE active=0;
                        QUERY PLAN
-----------------------------------------------------------
 Seq Scan on customer  (cost=0.00..16.49 rows=15 width=32)
   Filter: (active = 0)
(2 rows)

Заявката ΠΈΠ·Π²ΠΈΠΊΠ²Π° ΠΏΡŠΠ»Π½Π°Ρ‚Π° послСдоватСлност Π·Π° сканиранС Π½Π° Ρ‚Π°Π±Π»ΠΈΡ†Π° customer. НСка създадСм индСкс Π½Π° ΠΊΠΎΠ»ΠΎΠ½Π° active:

pagila=# CREATE INDEX idx_cust1 ON customer(active);
CREATE INDEX
pagila=# EXPLAIN SELECT email FROM customer WHERE active=0;
                                 QUERY PLAN
-----------------------------------------------------------------------------
 Index Scan using idx_cust1 on customer  (cost=0.28..12.29 rows=15 width=32)
   Index Cond: (active = 0)
(2 rows)

Помогна, послСдвалото сканиранС сС ΠΏΡ€Π΅Π²ΡŠΡ€Π½Π° Π² "index scan". Π’ΠΎΠ²Π° ΠΎΠ·Π½Π°Ρ‡Π°Π²Π°, Ρ‡Π΅ Postgres Ρ‰Π΅ сканира индСкса "idx_cust1", ΠΈ слСд Ρ‚ΠΎΠ²Π° ΠΏΡ€ΠΎΠ΄ΡŠΠ»ΠΆΠ΅Ρ‚Π΅ Π΄Π° Ρ‚ΡŠΡ€ΡΠΈΡ‚Π΅ Π² масата Π½Π° Ρ‚Π°Π±Π»ΠΈΡ†Π°Ρ‚Π°, Π·Π° Π΄Π° ΠΏΡ€ΠΎΡ‡Π΅Ρ‚Π΅Ρ‚Π΅ стойноститС Π½Π° Π΄Ρ€ΡƒΠ³ΠΈ ΠΊΠΎΠ»ΠΎΠ½ΠΈ (Π² Ρ‚ΠΎΠ·ΠΈ случай ΠΊΠΎΠ»ΠΎΠ½Π°Ρ‚Π° email), ΠΎΡ‚ ΠΊΠΎΠΈΡ‚ΠΎ сС Π½ΡƒΠΆΠ΄Π°Π΅ заявката.

ΠŸΠΎΠΊΡ€ΠΈΠ²Π°Ρ‰ΠΈΡ‚Π΅ индСкси са въвСдСни Π² PostgreSQL 11. Π’Π΅ Π²ΠΈ позволяват Π΄Π° Π²ΠΊΠ»ΡŽΡ‡ΠΈΡ‚Π΅ Π΅Π΄Π½Π° ΠΈΠ»ΠΈ ΠΏΠΎΠ²Π΅Ρ‡Π΅ Π΄ΠΎΠΏΡŠΠ»Π½ΠΈΡ‚Π΅Π»Π½ΠΈ ΠΊΠΎΠ»ΠΎΠ½ΠΈ Π² самия индСкс - Ρ‚Π΅Ρ…Π½ΠΈΡ‚Π΅ стойности сС ΡΡŠΡ…Ρ€Π°Π½ΡΠ²Π°Ρ‚ Π² Ρ…Ρ€Π°Π½ΠΈΠ»ΠΈΡ‰Π΅Ρ‚ΠΎ Π½Π° Π΄Π°Π½Π½ΠΈ Π½Π° индСкса.

Ако сС възползвамС ΠΎΡ‚ Ρ‚Π°Π·ΠΈ функция ΠΈ Π΄ΠΎΠ±Π°Π²ΠΈΠΌ ΠΈΠΌΠ΅ΠΉΠ» стойността Π²ΡŠΡ‚Ρ€Π΅ Π² индСкса, Ρ‚ΠΎΠ³Π°Π²Π° Postgres няма Π΄Π° ΠΈΠΌΠ° Π½ΡƒΠΆΠ΄Π° Π΄Π° Ρ‚ΡŠΡ€ΡΠΈ стойността Π² масата Π½Π° Ρ‚Π°Π±Π»ΠΈΡ†Π°Ρ‚Π°. email. Π”Π° Π²ΠΈΠ΄ΠΈΠΌ Π΄Π°Π»ΠΈ Ρ‚ΠΎΠ²Π° Ρ‰Π΅ ΠΏΡ€ΠΎΡ€Π°Π±ΠΎΡ‚ΠΈ:

pagila=# CREATE INDEX idx_cust2 ON customer(active) INCLUDE (email);
CREATE INDEX
pagila=# EXPLAIN SELECT email FROM customer WHERE active=0;
                                    QUERY PLAN
----------------------------------------------------------------------------------
 Index Only Scan using idx_cust2 on customer  (cost=0.28..12.29 rows=15 width=32)
   Index Cond: (active = 0)
(2 rows)

Β«Index Only Scan' Π½ΠΈ ΠΊΠ°Π·Π²Π°, Ρ‡Π΅ заявката сСга сС Π½ΡƒΠΆΠ΄Π°Π΅ само ΠΎΡ‚ индСкс, ΠΊΠΎΠΉΡ‚ΠΎ ΠΏΠΎΠΌΠ°Π³Π° Π΄Π° сС ΠΈΠ·Π±Π΅Π³Π½Π°Ρ‚ всички дискови I/O Π·Π° Ρ‡Π΅Ρ‚Π΅Π½Π΅ Π½Π° ΠΊΡƒΠΏΡ‡ΠΈΠ½Π°Ρ‚Π° Π½Π° Ρ‚Π°Π±Π»ΠΈΡ†Π°Ρ‚Π°.

ΠŸΠΎΠΊΡ€ΠΈΠ²Π°Ρ‰ΠΈΡ‚Π΅ индСкси Π² ΠΌΠΎΠΌΠ΅Π½Ρ‚Π° са Π½Π°Π»ΠΈΡ‡Π½ΠΈ само Π·Π° B-Π΄ΡŠΡ€Π²Π΅Ρ‚Π°. Π’ Ρ‚ΠΎΠ·ΠΈ случай ΠΎΠ±Π°Ρ‡Π΅ усилията Π·Π° ΠΏΠΎΠ΄Π΄Ρ€ΡŠΠΆΠΊΠ° Ρ‰Π΅ Π±ΡŠΠ΄Π°Ρ‚ ΠΏΠΎ-Π³ΠΎΠ»Π΅ΠΌΠΈ.

ИзползванС Π½Π° частични индСкси

ЧастичнитС индСкси индСксират само подмноТСство ΠΎΡ‚ Ρ€Π΅Π΄ΠΎΠ²Π΅Ρ‚Π΅ Π² Ρ‚Π°Π±Π»ΠΈΡ†Π°. Π’ΠΎΠ²Π° спСстява Ρ€Π°Π·ΠΌΠ΅Ρ€Π° Π½Π° индСкситС ΠΈ ΠΏΡ€Π°Π²ΠΈ сканиранСто ΠΏΠΎ-Π±ΡŠΡ€Π·ΠΎ.

Π”Π° ΠΊΠ°ΠΆΠ΅ΠΌ, Ρ‡Π΅ искамС Π΄Π° ΠΏΠΎΠ»ΡƒΡ‡ΠΈΠΌ списък с ΠΈΠΌΠ΅ΠΉΠ» адрСситС Π½Π° Π½Π°ΡˆΠΈΡ‚Π΅ ΠΊΠ»ΠΈΠ΅Π½Ρ‚ΠΈ Π² ΠšΠ°Π»ΠΈΡ„ΠΎΡ€Π½ΠΈΡ. Π˜ΡΠΊΠ°Π½Π΅Ρ‚ΠΎ Ρ‰Π΅ бъдС Ρ‚Π°ΠΊΠ°:

SELECT c.email FROM customer c
JOIN address a ON c.address_id = a.address_id
WHERE a.district = 'California';
which has a query plan that involves scanning both the tables that are joined:
pagila=# EXPLAIN SELECT c.email FROM customer c
pagila-# JOIN address a ON c.address_id = a.address_id
pagila-# WHERE a.district = 'California';
                              QUERY PLAN
----------------------------------------------------------------------
 Hash Join  (cost=15.65..32.22 rows=9 width=32)
   Hash Cond: (c.address_id = a.address_id)
   ->  Seq Scan on customer c  (cost=0.00..14.99 rows=599 width=34)
   ->  Hash  (cost=15.54..15.54 rows=9 width=4)
         ->  Seq Scan on address a  (cost=0.00..15.54 rows=9 width=4)
               Filter: (district = 'California'::text)
(6 rows)

Какво Ρ‰Π΅ Π½ΠΈ Π΄Π°Π΄Π°Ρ‚ ΠΎΠ±ΠΈΠΊΠ½ΠΎΠ²Π΅Π½ΠΈΡ‚Π΅ индСкси:

pagila=# CREATE INDEX idx_address1 ON address(district);
CREATE INDEX
pagila=# EXPLAIN SELECT c.email FROM customer c
pagila-# JOIN address a ON c.address_id = a.address_id
pagila-# WHERE a.district = 'California';
                                      QUERY PLAN
---------------------------------------------------------------------------------------
 Hash Join  (cost=12.98..29.55 rows=9 width=32)
   Hash Cond: (c.address_id = a.address_id)
   ->  Seq Scan on customer c  (cost=0.00..14.99 rows=599 width=34)
   ->  Hash  (cost=12.87..12.87 rows=9 width=4)
         ->  Bitmap Heap Scan on address a  (cost=4.34..12.87 rows=9 width=4)
               Recheck Cond: (district = 'California'::text)
               ->  Bitmap Index Scan on idx_address1  (cost=0.00..4.34 rows=9 width=0)
                     Index Cond: (district = 'California'::text)
(8 rows)

ΠΏΡ€Π΅Π³Π»Π΅ΠΆΠ΄Π°Π½Π΅ address Π΅ Π·Π°ΠΌΠ΅Π½Π΅Π½ ΠΎΡ‚ сканиранС Π½Π° индСкс idx_address1ΠΈ слСд Ρ‚ΠΎΠ²Π° сканира ΠΊΡƒΠΏΡ‡ΠΈΠ½Π°Ρ‚Π° address.

Въй ΠΊΠ°Ρ‚ΠΎ Ρ‚ΠΎΠ²Π° Π΅ чСста заявка ΠΈ трябва Π΄Π° сС ΠΎΠΏΡ‚ΠΈΠΌΠΈΠ·ΠΈΡ€Π°, ΠΌΠΎΠΆΠ΅ΠΌ Π΄Π° ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°ΠΌΠ΅ частичСн индСкс, ΠΊΠΎΠΉΡ‚ΠΎ индСксира само ΠΎΠ½Π΅Π·ΠΈ Ρ€Π΅Π΄ΠΎΠ²Π΅ с адрСси, Π² ΠΊΠΎΠΈΡ‚ΠΎ ΠΎΠΊΡ€ΡŠΠ³ΡŠΡ‚ β€˜California’:

pagila=# CREATE INDEX idx_address2 ON address(address_id) WHERE district='California';
CREATE INDEX
pagila=# EXPLAIN SELECT c.email FROM customer c
pagila-# JOIN address a ON c.address_id = a.address_id
pagila-# WHERE a.district = 'California';
                                           QUERY PLAN
------------------------------------------------------------------------------------------------
 Hash Join  (cost=12.38..28.96 rows=9 width=32)
   Hash Cond: (c.address_id = a.address_id)
   ->  Seq Scan on customer c  (cost=0.00..14.99 rows=599 width=34)
   ->  Hash  (cost=12.27..12.27 rows=9 width=4)
         ->  Index Only Scan using idx_address2 on address a  (cost=0.14..12.27 rows=9 width=4)
(5 rows)

Π‘Π΅Π³Π° заявката само Ρ‡Π΅Ρ‚Π΅ idx_address2 ΠΈ Π½Π΅ докосва масата address.

ИзползванС Π½Π° индСкси с мноТСство стойности

Някои ΠΊΠΎΠ»ΠΎΠ½ΠΈ Π·Π° индСксиранС ΠΌΠΎΠΆΠ΅ Π΄Π° Π½Π΅ ΡΡŠΠ΄ΡŠΡ€ΠΆΠ°Ρ‚ скаларСн Ρ‚ΠΈΠΏ Π΄Π°Π½Π½ΠΈ. Π’ΠΈΠΏΠΎΠ²Π΅ ΠΊΠΎΠ»ΠΎΠ½ΠΈ ΠΊΠ°Ρ‚ΠΎ jsonb, arrays ΠΈ tsvector ΡΡŠΠ΄ΡŠΡ€ΠΆΠ°Ρ‚ ΡΡŠΡΡ‚Π°Π²Π½ΠΈ ΠΈΠ»ΠΈ мноТСство стойности. Ако трябва Π΄Π° индСксиратС Ρ‚Π°ΠΊΠΈΠ²Π° ΠΊΠΎΠ»ΠΎΠ½ΠΈ, ΠΎΠ±ΠΈΠΊΠ½ΠΎΠ²Π΅Π½ΠΎ трябва Π΄Π° Ρ‚ΡŠΡ€ΡΠΈΡ‚Π΅ във всички ΠΎΡ‚Π΄Π΅Π»Π½ΠΈ стойности Π² Ρ‚Π΅Π·ΠΈ ΠΊΠΎΠ»ΠΎΠ½ΠΈ.

НСка сС ΠΎΠΏΠΈΡ‚Π°ΠΌΠ΅ Π΄Π° Π½Π°ΠΌΠ΅Ρ€ΠΈΠΌ заглавията Π½Π° всички Ρ„ΠΈΠ»ΠΌΠΈ, ΡΡŠΠ΄ΡŠΡ€ΠΆΠ°Ρ‰ΠΈ ΠΈΠ·Ρ€Π΅Π·ΠΊΠΈ ΠΎΡ‚ Π½Π΅ΡƒΡΠΏΠ΅ΡˆΠ½ΠΈ Π΄ΡƒΠ±Π»ΠΈ. Π’Π°Π±Π»ΠΈΡ†Π° film ΠΈΠΌΠ° тСкстова ΠΊΠΎΠ»ΠΎΠ½Π°, Π½Π°Ρ€Π΅Ρ‡Π΅Π½Π° special_features. Ако Ρ„ΠΈΠ»ΠΌΡŠΡ‚ ΠΈΠΌΠ° Ρ‚ΠΎΠ²Π° β€žΡΠΏΠ΅Ρ†ΠΈΠ°Π»Π½ΠΎ ΡΠ²ΠΎΠΉΡΡ‚Π²ΠΎβ€œ, Ρ‚ΠΎΠ³Π°Π²Π° ΠΊΠΎΠ»ΠΎΠ½Π°Ρ‚Π° ΡΡŠΠ΄ΡŠΡ€ΠΆΠ° Π΅Π»Π΅ΠΌΠ΅Π½Ρ‚Π° ΠΊΠ°Ρ‚ΠΎ тСкстов масив Behind The Scenes. Π—Π° Π΄Π° Ρ‚ΡŠΡ€ΡΠΈΠΌ всички Ρ‚Π°ΠΊΠΈΠ²Π° Ρ„ΠΈΠ»ΠΌΠΈ, трябва Π΄Π° ΠΈΠ·Π±Π΅Ρ€Π΅ΠΌ всички Ρ€Π΅Π΄ΠΎΠ²Π΅ с β€žΠ—Π°Π΄ ΠΊΡƒΠ»ΠΈΡΠΈΡ‚Π΅β€œ, ΠΊΠΎΠ³Π°Ρ‚ΠΎ ΠΊΠΎΠΉΡ‚ΠΎ ΠΈ Π΄Π° Π΅ стойности Π½Π° масива special_features:

SELECT title FROM film WHERE special_features @> '{"Behind The Scenes"}';

ΠžΠΏΠ΅Ρ€Π°Ρ‚ΠΎΡ€ Π·Π° Π²Π»Π°Π³Π°Π½Π΅ @> провСрява Π΄Π°Π»ΠΈ дясната страна Π΅ подмноТСство Π½Π° лявата страна.

План за заявка:

pagila=# EXPLAIN SELECT title FROM film
pagila-# WHERE special_features @> '{"Behind The Scenes"}';
                           QUERY PLAN
-----------------------------------------------------------------
 Seq Scan on film  (cost=0.00..67.50 rows=5 width=15)
   Filter: (special_features @> '{"Behind The Scenes"}'::text[])
(2 rows)

ΠšΠΎΠ΅Ρ‚ΠΎ изисква пълно сканиранС Π½Π° ΠΊΡƒΠΏΡ‡ΠΈΠ½Π° с Ρ†Π΅Π½Π° 67.

НСка Π΄Π° Π²ΠΈΠ΄ΠΈΠΌ Π΄Π°Π»ΠΈ ΠΎΠ±ΠΈΠΊΠ½ΠΎΠ²Π΅Π½ индСкс Π½Π° B-Π΄ΡŠΡ€Π²ΠΎ Π½ΠΈ ΠΏΠΎΠΌΠ°Π³Π°:

pagila=# CREATE INDEX idx_film1 ON film(special_features);
CREATE INDEX
pagila=# EXPLAIN SELECT title FROM film
pagila-# WHERE special_features @> '{"Behind The Scenes"}';
                           QUERY PLAN
-----------------------------------------------------------------
 Seq Scan on film  (cost=0.00..67.50 rows=5 width=15)
   Filter: (special_features @> '{"Behind The Scenes"}'::text[])
(2 rows)

Π˜Π½Π΄Π΅ΠΊΡΡŠΡ‚ Π΄ΠΎΡ€ΠΈ Π½Π΅ бСшС Ρ€Π°Π·Π³Π»Π΅Π΄Π°Π½. Π˜Π½Π΄Π΅ΠΊΡΡŠΡ‚ Π½Π° B-Π΄ΡŠΡ€Π²ΠΎΡ‚ΠΎ Π½Π΅ Π·Π½Π°Π΅ Π·Π° ΡΡŠΡ‰Π΅ΡΡ‚Π²ΡƒΠ²Π°Π½Π΅Ρ‚ΠΎ Π½Π° ΠΎΡ‚Π΄Π΅Π»Π½ΠΈ Π΅Π»Π΅ΠΌΠ΅Π½Ρ‚ΠΈ Π² индСксиранитС стойности.

ИмамС Π½ΡƒΠΆΠ΄Π° ΠΎΡ‚ GIN индСкс.

pagila=# CREATE INDEX idx_film2 ON film USING GIN(special_features);
CREATE INDEX
pagila=# EXPLAIN SELECT title FROM film
pagila-# WHERE special_features @> '{"Behind The Scenes"}';
                                QUERY PLAN
---------------------------------------------------------------------------
 Bitmap Heap Scan on film  (cost=8.04..23.58 rows=5 width=15)
   Recheck Cond: (special_features @> '{"Behind The Scenes"}'::text[])
   ->  Bitmap Index Scan on idx_film2  (cost=0.00..8.04 rows=5 width=0)
         Index Cond: (special_features @> '{"Behind The Scenes"}'::text[])
(4 rows)

Π˜Π½Π΄Π΅ΠΊΡΡŠΡ‚ GIN ΠΏΠΎΠ΄Π΄ΡŠΡ€ΠΆΠ° ΡΡŠΠΏΠΎΡΡ‚Π°Π²ΡΠ½Π΅ Π½Π° Π΅Π΄ΠΈΠ½ΠΈΡ‡Π½ΠΈ стойности срСщу индСксирани ΡΡŠΡΡ‚Π°Π²Π½ΠΈ стойности, ΠΊΠΎΠ΅Ρ‚ΠΎ Π²ΠΎΠ΄ΠΈ Π΄ΠΎ Ρ†Π΅Π½Π° Π½Π° ΠΏΠ»Π°Π½ Π·Π° заявка, която Π΅ ΠΏΠΎΠ²Π΅Ρ‡Π΅ ΠΎΡ‚ Π½Π°ΠΏΠΎΠ»ΠΎΠ²ΠΈΠ½Π°.

ΠžΡ‚ΡŠΡ€Π²Π΅Ρ‚Π΅ сС ΠΎΡ‚ Π΄ΡƒΠ±Π»ΠΈΡ€Π°Ρ‰ΠΈ сС индСкси

Π˜Π½Π΄Π΅ΠΊΡΠΈΡ‚Π΅ сС Π½Π°Ρ‚Ρ€ΡƒΠΏΠ²Π°Ρ‚ с Ρ‚Π΅Ρ‡Π΅Π½ΠΈΠ΅ Π½Π° Π²Ρ€Π΅ΠΌΠ΅Ρ‚ΠΎ ΠΈ понякога Π½ΠΎΠ² индСкс ΠΌΠΎΠΆΠ΅ Π΄Π° ΡΡŠΠ΄ΡŠΡ€ΠΆΠ° ΡΡŠΡ‰Π°Ρ‚Π° дСфиниция ΠΊΠ°Ρ‚ΠΎ Π΅Π΄ΠΈΠ½ ΠΎΡ‚ ΠΏΡ€Π΅Π΄ΠΈΡˆΠ½ΠΈΡ‚Π΅. ΠœΠΎΠΆΠ΅Ρ‚Π΅ Π΄Π° ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚Π΅ каталоТния ΠΈΠ·Π³Π»Π΅Π΄, Π·Π° Π΄Π° ΠΏΠΎΠ»ΡƒΡ‡ΠΈΡ‚Π΅ лСсни Π·Π° Ρ‡Π΅Ρ‚Π΅Π½Π΅ SQL Π΄Π΅Ρ„ΠΈΠ½ΠΈΡ†ΠΈΠΈ Π½Π° индСкси. pg_indexes. ΠœΠΎΠΆΠ΅Ρ‚Π΅ ΡΡŠΡ‰ΠΎ Ρ‚Π°ΠΊΠ° лСсно Π΄Π° Π½Π°ΠΌΠ΅Ρ€ΠΈΡ‚Π΅ ΠΈΠ΄Π΅Π½Ρ‚ΠΈΡ‡Π½ΠΈ опрСдСлСния:

 SELECT array_agg(indexname) AS indexes, replace(indexdef, indexname, '') AS defn
    FROM pg_indexes
GROUP BY defn
  HAVING count(*) > 1;
And here’s the result when run on the stock pagila database:
pagila=#   SELECT array_agg(indexname) AS indexes, replace(indexdef, indexname, '') AS defn
pagila-#     FROM pg_indexes
pagila-# GROUP BY defn
pagila-#   HAVING count(*) > 1;
                                indexes                                 |                                defn
------------------------------------------------------------------------+------------------------------------------------------------------
 {payment_p2017_01_customer_id_idx,idx_fk_payment_p2017_01_customer_id} | CREATE INDEX  ON public.payment_p2017_01 USING btree (customer_id
 {payment_p2017_02_customer_id_idx,idx_fk_payment_p2017_02_customer_id} | CREATE INDEX  ON public.payment_p2017_02 USING btree (customer_id
 {payment_p2017_03_customer_id_idx,idx_fk_payment_p2017_03_customer_id} | CREATE INDEX  ON public.payment_p2017_03 USING btree (customer_id
 {idx_fk_payment_p2017_04_customer_id,payment_p2017_04_customer_id_idx} | CREATE INDEX  ON public.payment_p2017_04 USING btree (customer_id
 {payment_p2017_05_customer_id_idx,idx_fk_payment_p2017_05_customer_id} | CREATE INDEX  ON public.payment_p2017_05 USING btree (customer_id
 {idx_fk_payment_p2017_06_customer_id,payment_p2017_06_customer_id_idx} | CREATE INDEX  ON public.payment_p2017_06 USING btree (customer_id
(6 rows)

БупСрмноТСство индСкси

МоТС Π΄Π° сС случи Π΄Π° ΠΏΠΎΠ»ΡƒΡ‡ΠΈΡ‚Π΅ ΠΌΠ½ΠΎΠ³ΠΎ индСкси, Π΅Π΄ΠΈΠ½ ΠΎΡ‚ ΠΊΠΎΠΈΡ‚ΠΎ индСксира надмноТСство ΠΎΡ‚ ΠΊΠΎΠ»ΠΎΠ½ΠΈ, ΠΊΠΎΠΈΡ‚ΠΎ индСксират Π΄Ρ€ΡƒΠ³ΠΈ индСкси. Π’ΠΎΠ²Π° ΠΌΠΎΠΆΠ΅ ΠΈΠ»ΠΈ Π½Π΅ ΠΌΠΎΠΆΠ΅ Π΄Π° Π΅ ΠΆΠ΅Π»Π°Ρ‚Π΅Π»Π½ΠΎ – надмноТСството ΠΌΠΎΠΆΠ΅ Π΄Π° Π΄ΠΎΠ²Π΅Π΄Π΅ Π΄ΠΎ сканиранС само Π½Π° индСкс, ΠΊΠΎΠ΅Ρ‚ΠΎ Π΅ Π΄ΠΎΠ±Ρ€Π΅, Π½ΠΎ ΠΌΠΎΠΆΠ΅ Π΄Π° Π·Π°Π΅ΠΌΠ° Ρ‚Π²ΡŠΡ€Π΄Π΅ ΠΌΠ½ΠΎΠ³ΠΎ място ΠΈΠ»ΠΈ заявката, която надмноТСството Π΅ ΠΏΡ€Π΅Π΄Π½Π°Π·Π½Π°Ρ‡Π΅Π½ΠΎ Π΄Π° ΠΎΠΏΡ‚ΠΈΠΌΠΈΠ·ΠΈΡ€Π°, Π²Π΅Ρ‡Π΅ Π½Π΅ сС ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°.

Ако трябва Π΄Π° Π°Π²Ρ‚ΠΎΠΌΠ°Ρ‚ΠΈΠ·ΠΈΡ€Π°Ρ‚Π΅ Π΄Π΅Ρ„ΠΈΠ½ΠΈΡ€Π°Π½Π΅Ρ‚ΠΎ Π½Π° Ρ‚Π°ΠΊΠΈΠ²Π° индСкси, ΠΌΠΎΠΆΠ΅Ρ‚Π΅ Π΄Π° Π·Π°ΠΏΠΎΡ‡Π½Π΅Ρ‚Π΅ с pg_index ΠΎΡ‚ масата pg_catalog.

НСизползвани индСкси

Π‘ Ρ€Π°Π·Π²ΠΈΡ‚ΠΈΠ΅Ρ‚ΠΎ Π½Π° прилоТСнията, ΠΊΠΎΠΈΡ‚ΠΎ ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚ Π±Π°Π·ΠΈ Π΄Π°Π½Π½ΠΈ, сС Ρ€Π°Π·Π²ΠΈΠ²Π°Ρ‚ ΠΈ заявкитС, ΠΊΠΎΠΈΡ‚ΠΎ ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚. Π˜Π½Π΄Π΅ΠΊΡΠΈΡ‚Π΅, Π΄ΠΎΠ±Π°Π²Π΅Π½ΠΈ ΠΏΠΎ-Ρ€Π°Π½ΠΎ, Π²Π΅Ρ‡Π΅ Π½Π΅ ΠΌΠΎΠ³Π°Ρ‚ Π΄Π° сС ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚ ΠΎΡ‚ Π½ΠΈΡ‚ΠΎ Π΅Π΄Π½Π° заявка. ВсСки ΠΏΡŠΡ‚, ΠΊΠΎΠ³Π°Ρ‚ΠΎ сС сканира индСкс, Ρ‚ΠΎΠΉ сС ΠΌΠ°Ρ€ΠΊΠΈΡ€Π° ΠΎΡ‚ ΠΌΠ΅Π½ΠΈΠ΄ΠΆΡŠΡ€Π° Π½Π° статистиката ΠΈ Π² ΠΈΠ·Π³Π»Π΅Π΄Π° Π½Π° систСмния ΠΊΠ°Ρ‚Π°Π»ΠΎΠ³ pg_stat_user_indexes ΠΌΠΎΠΆΠ΅Ρ‚Π΅ Π΄Π° Π²ΠΈΠ΄ΠΈΡ‚Π΅ стойността idx_scan, ΠΊΠΎΠ΅Ρ‚ΠΎ Π΅ ΠΊΡƒΠΌΡƒΠ»Π°Ρ‚ΠΈΠ²Π΅Π½ брояч. ΠŸΡ€ΠΎΡΠ»Π΅Π΄ΡΠ²Π°Π½Π΅Ρ‚ΠΎ Π½Π° Ρ‚Π°Π·ΠΈ стойност Π·Π° ΠΎΠΏΡ€Π΅Π΄Π΅Π»Π΅Π½ ΠΏΠ΅Ρ€ΠΈΠΎΠ΄ ΠΎΡ‚ Π²Ρ€Π΅ΠΌΠ΅ (Π΄Π° Ρ€Π΅Ρ‡Π΅ΠΌ мСсСц) Ρ‰Π΅ Π΄Π°Π΄Π΅ Π΄ΠΎΠ±Ρ€Π° прСдстава ΠΊΠΎΠΈ индСкси Π½Π΅ сС ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚ ΠΈ ΠΌΠΎΠ³Π°Ρ‚ Π΄Π° Π±ΡŠΠ΄Π°Ρ‚ ΠΏΡ€Π΅ΠΌΠ°Ρ…Π½Π°Ρ‚ΠΈ.

Π•Ρ‚ΠΎ заявка Π·Π° ΠΏΠΎΠ»ΡƒΡ‡Π°Π²Π°Π½Π΅ Π½Π° тСкущия Π±Ρ€ΠΎΠΉ сканирания Π½Π° всички индСкси Π² схСмата β€˜public’:

SELECT relname, indexrelname, idx_scan
FROM   pg_catalog.pg_stat_user_indexes
WHERE  schemaname = 'public';
with output like this:
pagila=# SELECT relname, indexrelname, idx_scan
pagila-# FROM   pg_catalog.pg_stat_user_indexes
pagila-# WHERE  schemaname = 'public'
pagila-# LIMIT  10;
    relname    |    indexrelname    | idx_scan
---------------+--------------------+----------
 customer      | customer_pkey      |    32093
 actor         | actor_pkey         |     5462
 address       | address_pkey       |      660
 category      | category_pkey      |     1000
 city          | city_pkey          |      609
 country       | country_pkey       |      604
 film_actor    | film_actor_pkey    |        0
 film_category | film_category_pkey |        0
 film          | film_pkey          |    11043
 inventory     | inventory_pkey     |    16048
(10 rows)

ΠŸΠΎΠ²Ρ‚ΠΎΡ€Π½ΠΎ ΠΈΠ·Π³Ρ€Π°ΠΆΠ΄Π°Π½Π΅ Π½Π° индСкси с ΠΏΠΎ-ΠΌΠ°Π»ΠΊΠΎ Π·Π°ΠΊΠ»ΡŽΡ‡Π²Π°Π½ΠΈΡ

Π˜Π½Π΄Π΅ΠΊΡΠΈΡ‚Π΅ чСсто сС нуТдаят ΠΎΡ‚ ΠΏΠΎΠ²Ρ‚ΠΎΡ€Π½ΠΎ ΠΈΠ·Π³Ρ€Π°ΠΆΠ΄Π°Π½Π΅, Π½Π°ΠΏΡ€ΠΈΠΌΠ΅Ρ€ ΠΊΠΎΠ³Π°Ρ‚ΠΎ сС раздуят, ΠΈ ΠΏΠΎΠ²Ρ‚ΠΎΡ€Π½ΠΎΡ‚ΠΎ ΠΈΠ·Π³Ρ€Π°ΠΆΠ΄Π°Π½Π΅ ΠΌΠΎΠΆΠ΅ Π΄Π° ускори сканиранСто. Π‘ΡŠΡ‰ΠΎ Ρ‚Π°ΠΊΠ° индСкситС ΠΌΠΎΠ³Π°Ρ‚ Π΄Π° сС поврСдят. ΠŸΡ€ΠΎΠΌΡΠ½Π°Ρ‚Π° Π½Π° ΠΏΠ°Ρ€Π°ΠΌΠ΅Ρ‚Ρ€ΠΈΡ‚Π΅ Π½Π° индСкса ΠΌΠΎΠΆΠ΅ ΡΡŠΡ‰ΠΎ Π΄Π° изисква ΠΏΠΎΠ²Ρ‚ΠΎΡ€Π½ΠΎΡ‚ΠΎ ΠΌΡƒ ΠΈΠ·Π³Ρ€Π°ΠΆΠ΄Π°Π½Π΅.

Π Π°Π·Ρ€Π΅ΡˆΠ΅Ρ‚Π΅ ΡΡŠΠ·Π΄Π°Π²Π°Π½Π΅Ρ‚ΠΎ Π½Π° ΠΏΠ°Ρ€Π°Π»Π΅Π»Π΅Π½ индСкс

Π’ PostgreSQL 11 ΡΡŠΠ·Π΄Π°Π²Π°Π½Π΅Ρ‚ΠΎ Π½Π° индСкс Π½Π° B-Tree Π΅ Π΅Π΄Π½ΠΎΠ²Ρ€Π΅ΠΌΠ΅Π½Π½ΠΎ. Π—Π° Π΄Π° сС ускори ΠΏΡ€ΠΎΡ†Π΅ΡΡŠΡ‚ Π½Π° създаванС, ΠΌΠΎΠ³Π°Ρ‚ Π΄Π° сС ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚ няколко ΠΏΠ°Ρ€Π°Π»Π΅Π»Π½ΠΈ Ρ€Π°Π±ΠΎΡ‚Π½ΠΈΡ†ΠΈ. Π£Π²Π΅Ρ€Π΅Ρ‚Π΅ сС ΠΎΠ±Π°Ρ‡Π΅, Ρ‡Π΅ Ρ‚Π΅Π·ΠΈ ΠΎΠΏΡ†ΠΈΠΈ Π·Π° конфигурация са Π·Π°Π΄Π°Π΄Π΅Π½ΠΈ ΠΏΡ€Π°Π²ΠΈΠ»Π½ΠΎ:

SET max_parallel_workers = 32;
SET max_parallel_maintenance_workers = 16;

БтойноститС ΠΏΠΎ ΠΏΠΎΠ΄Ρ€Π°Π·Π±ΠΈΡ€Π°Π½Π΅ са Ρ‚Π²ΡŠΡ€Π΄Π΅ ΠΌΠ°Π»ΠΊΠΈ. Π’ идСалния случай Ρ‚Π΅Π·ΠΈ числа трябва Π΄Π° нарастват Π·Π°Π΅Π΄Π½ΠΎ с броя Π½Π° процСсорнитС ядра. ΠŸΡ€ΠΎΡ‡Π΅Ρ‚Π΅Ρ‚Π΅ ΠΏΠΎΠ²Π΅Ρ‡Π΅ Π² докумСнтация.

БъздаванС Π½Π° Ρ„ΠΎΠ½ΠΎΠ² индСкс

ΠœΠΎΠΆΠ΅Ρ‚Π΅ Π΄Π° ΡΡŠΠ·Π΄Π°Π΄Π΅Ρ‚Π΅ индСкс във Ρ„ΠΎΠ½ΠΎΠ² Ρ€Π΅ΠΆΠΈΠΌ, ΠΊΠ°Ρ‚ΠΎ ΠΈΠ·ΠΏΠΎΠ»Π·Π²Π°Ρ‚Π΅ опцията CONCURRENTLY ΠΊΠΎΠΌΠ°Π½Π΄ΠΈ CREATE INDEX:

pagila=# CREATE INDEX CONCURRENTLY idx_address1 ON address(district);
CREATE INDEX

Π’Π°Π·ΠΈ ΠΏΡ€ΠΎΡ†Π΅Π΄ΡƒΡ€Π° Π·Π° създаванС Π½Π° индСкс сС Ρ€Π°Π·Π»ΠΈΡ‡Π°Π²Π° ΠΎΡ‚ ΠΎΠ±ΠΈΡ‡Π°ΠΉΠ½Π°Ρ‚Π° ΠΏΠΎ Ρ‚ΠΎΠ²Π°, Ρ‡Π΅ Π½Π΅ изисква Π·Π°ΠΊΠ»ΡŽΡ‡Π²Π°Π½Π΅ Π½Π° Ρ‚Π°Π±Π»ΠΈΡ†Π°Ρ‚Π° ΠΈ слСдоватСлно Π½Π΅ Π±Π»ΠΎΠΊΠΈΡ€Π° ΠΎΠΏΠ΅Ρ€Π°Ρ†ΠΈΠΈΡ‚Π΅ Π·Π° запис. ΠžΡ‚ Π΄Ρ€ΡƒΠ³Π° страна ΠΎΡ‚Π½Π΅ΠΌΠ° ΠΏΠΎΠ²Π΅Ρ‡Π΅ Π²Ρ€Π΅ΠΌΠ΅ ΠΈ ΠΈΠ·Ρ€Π°Π·Ρ…ΠΎΠ΄Π²Π° ΠΏΠΎΠ²Π΅Ρ‡Π΅ рСсурси.

Postgres прСдоставя ΠΌΠ½ΠΎΠ³ΠΎ Π³ΡŠΠ²ΠΊΠ°Π²ΠΎΡΡ‚ Π·Π° създаванС Π½Π° индСкси ΠΈ Π½Π°Ρ‡ΠΈΠ½ΠΈ Π·Π° Ρ€Π΅ΡˆΠ°Π²Π°Π½Π΅ Π½Π° всякакви спСциални случаи, ΠΊΠ°ΠΊΡ‚ΠΎ ΠΈ Π½Π°Ρ‡ΠΈΠ½ΠΈ Π·Π° ΡƒΠΏΡ€Π°Π²Π»Π΅Π½ΠΈΠ΅ Π½Π° Π±Π°Π·Π°Ρ‚Π° Π΄Π°Π½Π½ΠΈ, Π² случай Ρ‡Π΅ ΠΏΡ€ΠΈΠ»ΠΎΠΆΠ΅Π½ΠΈΠ΅Ρ‚ΠΎ Π²ΠΈ растС Сксплозивно. НадявамС сС, Ρ‡Π΅ Ρ‚Π΅Π·ΠΈ ΡΡŠΠ²Π΅Ρ‚ΠΈ Ρ‰Π΅ Π²ΠΈ ΠΏΠΎΠΌΠΎΠ³Π½Π°Ρ‚ Π΄Π° ΠΏΠΎΠ»ΡƒΡ‡ΠΈΡ‚Π΅ Π²Π°ΡˆΠΈΡ‚Π΅ заявки Π±ΡŠΡ€Π·ΠΎ ΠΈ Π²Π°ΡˆΠ°Ρ‚Π° Π±Π°Π·Π° Π΄Π°Π½Π½ΠΈ Π³ΠΎΡ‚ΠΎΠ²Π° Π·Π° ΠΌΠ°Ρ‰Π°Π±ΠΈΡ€Π°Π½Π΅.

Π˜Π·Ρ‚ΠΎΡ‡Π½ΠΈΠΊ: www.habr.com

ДобавянС Π½Π° Π½ΠΎΠ² ΠΊΠΎΠΌΠ΅Π½Ρ‚Π°Ρ€