在线精品99_中国九九盗摄偷拍偷看_91免费版在线观看_91.app_91高清视频在线_99热最新网站

怎么正确使用PostgreSQL中的OR

151次阅读
没有评论

共计 4117 个字符,预计需要花费 11 分钟才能阅读完成。

本篇内容介绍了“怎么正确使用 PostgreSQL 中的 OR”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让丸趣 TV 小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

在 SQL 语句中,对 OR 使用不当可能会导致较差的查询效率。这并不意味着不能用 OR 而是在使用 OR 时需考虑可能存在的性能问题。
测试数据:

DROP TABLE a;
CREATE TABLE a(id integer NOT NULL, a_val text NOT NULL);
INSERT INTO a
 SELECT i, md5(i::text)
 FROM generate_series(1, 1000000) i;
DROP TABLE b; 
CREATE TABLE b(id integer NOT NULL, b_val text NOT NULL);
INSERT INTO b
 SELECT i, md5(i::text)
 FROM generate_series(1, 1000000) i;
ALTER TABLE a ADD PRIMARY KEY (id);
ALTER TABLE b ADD PRIMARY KEY (id);
ALTER TABLE b ADD FOREIGN KEY (id) REFERENCES a;
VACUUM (ANALYZE) a;
VACUUM (ANALYZE) b;

OR vs IN
条件语句 p1 OR p2,如可以考虑使用 IN 来改写,比如:

[local:/data/pg12]:5432 pg12@testdb=# EXPLAIN verbose
SELECT id FROM a
WHERE id = 42
 OR id = 4711;
 QUERY PLAN 
---------------------------------------------------------------------------
 Bitmap Heap Scan on public.a (cost=8.87..16.80 rows=2 width=4)
 Output: id
 Recheck Cond: ((a.id = 42) OR (a.id = 4711))
 -  BitmapOr (cost=8.87..8.87 rows=2 width=0)
 -  Bitmap Index Scan on a_pkey (cost=0.00..4.43 rows=1 width=0)
 Index Cond: (a.id = 42)
 -  Bitmap Index Scan on a_pkey (cost=0.00..4.43 rows=1 width=0)
 Index Cond: (a.id = 4711)
(8 rows)
[local:/data/pg12]:5432 pg12@testdb=# 
[local:/data/pg12]:5432 pg12@testdb=# EXPLAIN verbose
SELECT id FROM a
WHERE id in (42,4711);
 QUERY PLAN 
----------------------------------------------------------------------------
 Index Only Scan using a_pkey on public.a (cost=0.42..8.88 rows=2 width=4)
 Output: id
 Index Cond: (a.id = ANY ( {42,4711} ::integer[]))
(3 rows)
[local:/data/pg12]:5432 pg12@testdb=#

使用 OR 操作符,PG 优化器走的是 Bitmap Index Scan,使用 IN,优化器选择的路径是 Index Only Scan,相对于 Bitmap Index Scan 少了 Bitmap 的建立,成本自然要低不少。

OR and Join
在 Join 场景中,如果在参与 join 的表上都存在查询条件然后在 where 子句中应用 OR 关联,那么优化器会选择 a 和 b 连接然后使用 Filter 过滤,由于先进行 join 而没有进行谓词下推,因此为了得到 1 行而 filter 了 999999 行,代价巨大。

[local:/data/pg12]:5432 pg12@testdb=# EXPLAIN verbose 
SELECT id, a.a_val, b.b_val
FROM a JOIN b USING (id)
WHERE a.id = 42
 OR b.id = 42;
 QUERY PLAN 
---------------------------------------------------------------------------------------------
 Gather (cost=21965.00..45327.62 rows=2 width=70)
 Output: a.id, a.a_val, b.b_val
 Workers Planned: 2
 -  Parallel Hash Join (cost=20965.00..44327.42 rows=1 width=70)
 Output: a.id, a.a_val, b.b_val
 Inner Unique: true
 Hash Cond: (a.id = b.id)
 Join Filter: ((a.id = 42) OR (b.id = 42))
 -  Parallel Seq Scan on public.a (cost=0.00..12500.67 rows=416667 width=37)
 Output: a.id, a.a_val
 -  Parallel Hash (cost=12500.67..12500.67 rows=416667 width=37)
 Output: b.b_val, b.id
 -  Parallel Seq Scan on public.b (cost=0.00..12500.67 rows=416667 width=37)
 Output: b.b_val, b.id
(14 rows)

在这种情况下,可以通过使用 UNION 来关联两个 JOIN 提升性能

[local:/data/pg12]:5432 pg12@testdb=# EXPLAIN verbose
pg12@testdb-# SELECT id, a.a_val, b.b_val
pg12@testdb-# FROM a JOIN b USING (id)
pg12@testdb-# WHERE a.id = 42
pg12@testdb-# UNION
pg12@testdb-# SELECT id, a.a_val, b.b_val
pg12@testdb-# FROM a JOIN b USING (id)
pg12@testdb-# WHERE b.id = 42
pg12@testdb-# ;
 QUERY PLAN 
----------------------------------------------------------------------------------------------------
 Unique (cost=33.83..33.85 rows=2 width=68)
 Output: a.id, a.a_val, b.b_val
 -  Sort (cost=33.83..33.84 rows=2 width=68)
 Output: a.id, a.a_val, b.b_val
 Sort Key: a.id, a.a_val, b.b_val
 -  Append (cost=0.85..33.82 rows=2 width=68)
 -  Nested Loop (cost=0.85..16.90 rows=1 width=70)
 Output: a.id, a.a_val, b.b_val
 -  Index Scan using a_pkey on public.a (cost=0.42..8.44 rows=1 width=37)
 Output: a.id, a.a_val
 Index Cond: (a.id = 42)
 -  Index Scan using b_pkey on public.b (cost=0.42..8.44 rows=1 width=37)
 Output: b.id, b.b_val
 Index Cond: (b.id = 42)
 -  Nested Loop (cost=0.85..16.90 rows=1 width=70)
 Output: a_1.id, a_1.a_val, b_1.b_val
 -  Index Scan using a_pkey on public.a a_1 (cost=0.42..8.44 rows=1 width=37)
 Output: a_1.id, a_1.a_val
 Index Cond: (a_1.id = 42)
 -  Index Scan using b_pkey on public.b b_1 (cost=0.42..8.44 rows=1 width=37)
 Output: b_1.id, b_1.b_val
 Index Cond: (b_1.id = 42)
(22 rows)
[local:/data/pg12]:5432 pg12@testdb=#

两个子连接选择了成本最低的 NL join,总成本是原来 SQL 语句成本的 0.1% 都不到,差了 3 个数量级。

“怎么正确使用 PostgreSQL 中的 OR”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注丸趣 TV 网站,丸趣 TV 小编将为大家输出更多高质量的实用文章!

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2023-07-26发表,共计4117字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)
主站蜘蛛池模板: 国产精品久久无码不卡黑寡妇 | 亚洲欧美在线精品一区二区 | 亚洲性啪啪无码av天堂 | 国产成人精品无码片区在线观看 | 一区二区不卡 | 亚洲色一区二区三区四区 | 国产色视频在线 | 亚洲av成人无码精品直播在线 | 国产av旡码专区亚洲av苍井空 | 一级免费毛片 | 久久一日本道色综合久久m 久久一日本综合色鬼综合色 | www视频在线观看免费 | 深夜看片在线观看18 | 午夜a一级毛片一.成 | 精品在线视频观看 | 精精国产xxxx视频在线 | 色综合天天综合狠狠爱 | 成人在线h| 久久一本一区二区三区 | 无码丰满熟妇一区二区 | 国产69精品久久久久99 | 在线观看h片 | 亚洲色图 激情小说 | 免费成年人视频网站 | 日韩草逼视频 | 亚洲日韩欧美一区二区三区 | 欧美成人午夜毛片免费影院 | 成人免费毛片视频 | 日本午夜影院 | 亚洲女同成av人片在线观看 | 午夜精品一区二区三区的区别 | 日本三级吃奶头添泬 | 欧美亚洲国产精品久久蜜芽 | 国产精品人妻一区二区三区四 | 久久人妻少妇嫩草av蜜桃 | 中文字幕日韩一区二区三区不卡 | 国产成人精品亚洲日本语言 | 亚洲av无码专区在线电影 | 国产免费久久精品久久久 | 欧美亚洲性色影视在线 | 免费看午夜高清性色生活片 |