使用 varchar_pattern_ops 索引而不是 pkey 索引的 Django-Postgres WHERE 查询

Django-Postgres WHERE query using varchar_pattern_ops index instead of pkey index

我有一个 Django-Postgres 设置 table -

class User(models.Model):
    id = models.CharField(max_length=255, primary_key=True)

运行 迁移在该字段上创建了两个索引(这是 Django 自动执行的操作,正如我从 运行ning sqlmigrate 中检查的那样)——一个索引用于 pkey,一个索引用于 varchar_pattern_ops -

\d+ "user";

Column|            Type          | Modifiers | Storage  | Stats target | Description 
------+--------------------------+-----------+----------+--------------+-------------
 id   |  character varying(255)  | not null  | extended |              | 

Indexes:
"user_pkey" PRIMARY KEY, btree (id)
"user_id_90845346_like" btree (id varchar_pattern_ops)

据我了解,如果我运行这个查询

select * from "user" where id='id1234';

应该使用user_pkey。相反,它使用 user_id_90845346_like.

explain analyze select * from "user" where id='id1234';

 Index Scan using "user_id_90845346_like" on "user"  (cost=0.41..8.43 rows=1 width=770) (actual time=0.033..0.0
33 rows=0 loops=1)
   Index Cond: ((id)::text = 'id1234'::text)
 Planning time: 1.335 ms
 Execution time: 0.072 ms
(4 rows)

我也没有看到任何强制 Postgres 使用索引的选项,但我真正想知道的是为什么 = 搜索不使用主键。 like text% 搜索不应该使用 varchar_pattern_ops 索引吗?

postgres 驱动程序总是会选择 varchar_pattern_ops 索引,如果它存在于您索引的列是 varchar 列或其某些变体的情况下。简而言之,因为要建立索引的列包含字符串,所以驱动程序会选择最适合字符串的索引(如果可用)。如果您将主键存储为整数,驱动程序将使用 btree 索引。