首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >postgresql :用数组自联接

postgresql :用数组自联接
EN

Stack Overflow用户
提问于 2018-12-10 12:26:16
回答 2查看 405关注 0票数 0

我的问题是如何为下面的用例生成Postgres SQL查询

Approach#1

我有一个如下所示的表,其中我在不同类型(a、b、c、d)中生成相同的uuid,类似于映射不同类型。

代码语言:javascript
复制
+----+------+-------------+
| id | type | master_guid |
+----+------+-------------+
|  1 | a    | uuid-1      |
|  2 | a    | uuid-2      |
|  3 | a    | uuid-3      |
|  4 | a    | uuid-4      |
|  5 | a    | uuid-5      |
|  6 | b    | uuid-1      |
|  7 | b    | uuid-2      |
|  8 | b    | uuid-3      |
|  9 | b    | uuid-6      |
| 10 | c    | uuid-1      |
| 11 | c    | uuid-2      |
| 12 | c    | uuid-3      |
| 13 | c    | uuid-6      |
| 14 | c    | uuid-7      |
| 15 | d    | uuid-6      |
| 16 | d    | uuid-2      |
+----+------+-------------+

Approach#2

我创建了两个表,以便输入id,然后将id输入到master_guid,如下所示

table1:

代码语言:javascript
复制
+----+------+
| id | type |
+----+------+
|  1 | a    |
|  2 | a    |
|  3 | a    |
|  4 | a    |
|  5 | a    |
|  6 | b    |
|  7 | b    |
|  8 | b    |
|  9 | b    |
| 10 | c    |
| 11 | c    |
| 12 | c    |
| 13 | c    |
| 14 | c    |
| 15 | d    |
| 16 | d    |
+----+------+

table2

代码语言:javascript
复制
+----+-------------+
| id | master_guid |
+----+-------------+
|  1 | uuid-1      |
|  2 | uuid-2      |
|  3 | uuid-3      |
|  4 | uuid-4      |
|  5 | uuid-5      |
|  6 | uuid-1      |
|  7 | uuid-2      |
|  8 | uuid-3      |
|  9 | uuid-6      |
| 10 | uuid-1      |
| 11 | uuid-2      |
| 12 | uuid-3      |
| 13 | uuid-6      |
| 14 | uuid-7      |
| 15 | uuid-6      |
| 16 | uuid-2      |
+----+-------------+

我希望在这两种方法中都能得到如下输出:

代码语言:javascript
复制
+----+------+--------+------------+
| id | type |  uuid  | mapped_ids |
+----+------+--------+------------+
|  1 | a    | uuid-1 | [6,10]     |
|  2 | a    | uuid-2 | [7,11]     |
|  3 | a    | uuid-3 | [8,12]     |
|  4 | a    | uuid-4 | null       |
|  5 | a    | uuid-5 | null       |
+----+------+--------+------------+

我尝试过在I和基于uuid的分组上使用array_agg进行自联接,但是无法获得所需的输出。

使用下面的查询填充数据:

Approach#1

代码语言:javascript
复制
insert into table1 values 
(1,'a','uuid-1'),
(2,'a','uuid-2'),
(3,'a','uuid-3'),
(4,'a','uuid-4'),
(5,'a','uuid-5'),
(6,'b','uuid-1'),
(7,'b','uuid-2'),
(8,'b','uuid-3'),
(9,'b','uuid-6'),
(10,'c','uuid-1'),
(11,'c','uuid-2'),
(12,'c','uuid-3'),
(13,'c','uuid-6'),
(14,'c','uuid-7'),
(15,'d','uuid-6'),
(16,'d','uuid-2')

Approach#2

代码语言:javascript
复制
insert into table1 values 
(1,'a'),
(2,'a'),
(3,'a'),
(4,'a'),
(5,'a'),
(6,'b'),
(7,'b'),
(8,'b'),
(9,'b'),
(10,'c'),
(11,'c'),
(12,'c'),
(13,'c'),
(14,'c'),
(15,'d'),
(16,'d')

insert into table2 values 
(1,'uuid-1'),
(2,'uuid-2'),
(3,'uuid-3'),
(4,'uuid-4'),
(5,'uuid-5'),
(6,'uuid-1'),
(7,'uuid-2'),
(8,'uuid-3'),
(9,'uuid-6'),
(10,'uuid-1'),
(11,'uuid-2'),
(12,'uuid-3'),
(13,'uuid-6'),
(14,'uuid-7'),
(15,'uuid-6'),
(16,'uuid-2')
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2018-12-10 12:53:59

试试这个:

代码语言:javascript
复制
select
  t1.id, t1.type, t1.master_guid, array_agg (distinct t2.id)
from
  table1 t1
  left join table1 t2 on
    t1.master_guid = t2.master_guid and
    t1.id != t2.id
group by
  t1.id, t1.type, t1.master_guid

我没有想出与你所列出的结果完全相同的结果,但我认为,也许你的预期是错误的,或者我的错误只是一个小小的错误。不管是哪种方式,一个潜在的起点。

-编辑--

对于方法2,我认为只需向Table2添加一个内部连接即可获得GUID:

代码语言:javascript
复制
select
  t1.id, t1.type, t2.master_guid,
  array_agg (t2a.id)
from
  table1 t1
  join table2 t2 on t1.id = t2.id
  left join table2 t2a on
    t2.master_guid = t2a.master_guid and
    t2a.id != t1.id
where
  t1.type = 'a'
group by
  t1.id, t1.type, t2.master_guid
票数 0
EN

Stack Overflow用户

发布于 2018-12-10 12:53:58

demo: db<>fiddle

使用ARRAY_AGG使您可以聚合每个组的id(在您的例子中,组是您的uuid)

代码语言:javascript
复制
SELECT 
    id, type, master_guid as uuid, 
    array_agg(id) OVER (PARTITION BY master_guid) as mapped_ids
FROM table1
ORDER BY id

结果:

代码语言:javascript
复制
| id | type |   uuid | mapped_ids |
|----|------|--------|------------|
|  1 |    a | uuid-1 |     10,6,1 |
|  2 |    a | uuid-2 |  16,2,7,11 |
|  3 |    a | uuid-3 |     8,3,12 |
|  4 |    a | uuid-4 |          4 |
|  5 |    a | uuid-5 |          5 |
|  6 |    b | uuid-1 |     10,6,1 |
|  7 |    b | uuid-2 |  16,2,7,11 |
|  8 |    b | uuid-3 |     8,3,12 |
|  9 |    b | uuid-6 |    15,13,9 |
| 10 |    c | uuid-1 |     10,6,1 |
| 11 |    c | uuid-2 |  16,2,7,11 |
| 12 |    c | uuid-3 |     8,3,12 |
| 13 |    c | uuid-6 |    15,13,9 |
| 14 |    c | uuid-7 |         14 |
| 15 |    d | uuid-6 |    15,13,9 |
| 16 |    d | uuid-2 |  16,2,7,11 |

这些数组当前还包含当前行的id (mapped_ids of id = 1包含1)。可以通过使用array_remove删除此元素来纠正这一问题。

代码语言:javascript
复制
SELECT 
    id, type, master_guid as uuid,  
    array_remove(array_agg(id) OVER (PARTITION BY master_guid), id) as mapped_ids
FROM table1
ORDER BY id

结果:

代码语言:javascript
复制
| id | type |   uuid | mapped_ids |
|----|------|--------|------------|
|  1 |    a | uuid-1 |       10,6 |
|  2 |    a | uuid-2 |    16,7,11 |
|  3 |    a | uuid-3 |       8,12 |
|  4 |    a | uuid-4 |            |
|  5 |    a | uuid-5 |            |
|  6 |    b | uuid-1 |       10,1 |
|  7 |    b | uuid-2 |    16,2,11 |
|  8 |    b | uuid-3 |       3,12 |
|  9 |    b | uuid-6 |      15,13 |
| 10 |    c | uuid-1 |        6,1 |
| 11 |    c | uuid-2 |     16,2,7 |
| 12 |    c | uuid-3 |        8,3 |
| 13 |    c | uuid-6 |       15,9 |
| 14 |    c | uuid-7 |            |
| 15 |    d | uuid-6 |       13,9 |
| 16 |    d | uuid-2 |     2,7,11 |

例如,现在id=4包含一个空数组,而不是NULL值。这可以通过使用NULLIF函数来实现。如果两个参数相等,这将给出NULL,否则它会给出第一个参数。

代码语言:javascript
复制
SELECT 
    id, type, master_guid as uuid,  
    NULLIF(
        array_remove(array_agg(id) OVER (PARTITION BY master_guid), id), 
        '{}'::int[]
    ) as mapped_ids 
FROM table1
ORDER BY id

结果:

代码语言:javascript
复制
| id | type |   uuid | mapped_ids |
|----|------|--------|------------|
|  1 |    a | uuid-1 |       10,6 |
|  2 |    a | uuid-2 |    16,7,11 |
|  3 |    a | uuid-3 |       8,12 |
|  4 |    a | uuid-4 |     (null) |
|  5 |    a | uuid-5 |     (null) |
|  6 |    b | uuid-1 |       10,1 |
|  7 |    b | uuid-2 |    16,2,11 |
|  8 |    b | uuid-3 |       3,12 |
|  9 |    b | uuid-6 |      15,13 |
| 10 |    c | uuid-1 |        6,1 |
| 11 |    c | uuid-2 |     16,2,7 |
| 12 |    c | uuid-3 |        8,3 |
| 13 |    c | uuid-6 |       15,9 |
| 14 |    c | uuid-7 |     (null) |
| 15 |    d | uuid-6 |       13,9 |
| 16 |    d | uuid-2 |     2,7,11 |
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/53705635

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档