将多行合并为单行 PostgreSQL

Nil*_*ade 7 postgresql merge join

我有一个名为deliveries 的主表,它与deliveries_languages 作为dl、deliveries_markets dm 和deliveries_tags dt 具有一对多的关系,其中deliveries_id 作为外键。这 3 个表分别与语言、市场和标签具有一一对应的关系。另外,deliveries, table 与公司有一对一的关系,并且有 company_is 作为外键。以下是我写的一个查询:

SELECT deliveries.*, languages.display_name, markets.default_name, tags.default_name, companies.name
FROM deliveries
JOIN deliveries_languages dl ON dl.delivery_id = deliveries.id 
JOIN deliveries_markets dm ON dm.delivery_id = deliveries.id 
JOIN deliveries_tags dt ON dt.delivery_id = deliveries.id 
JOIN languages ON languages.id = dl.language_id 
JOIN markets ON markets.id = dm.market_id 
JOIN tags ON tags.id = dt.tag_id 
JOIN companies ON companies.id = deliveries.company_id 
WHERE 
deliveries.name ILIKE '%new%' AND 
deliveries.created_by = '5f331347-fb58-4f63-bcf0-702f132f97c5' AND 
deliveries.deleted_at IS NULL 
LIMIT 10
Run Code Online (Sandbox Code Playgroud)

在这里,我得到了多余的 delivery_ids,因为对于每个 delivery_id 都有多种语言、市场和标签。我想对不同的 delivery_ids 使用限制,同时,我希望将这些多种语言、市场和标签分组并填充在单行中。

目前它看起来像:

delivery_id | name |languages | markets  | tags
------------|------|----------|----------|-----------
1           | d1   |en        | au       | tag1
1           | d1   |de        | sw       | tag2
2           | d2   |en        | au       | tag1
2           | d2   |de        | sw       | tag2
3           | d3   |en        | au       | tag1
3           | d3   |de        | sw       | tag2
Run Code Online (Sandbox Code Playgroud)

有什么办法可以让数据看起来像下面这样:

delivery_id | name |languages | markets  | tags
------------|------|----------|----------|-----------
1           | d1   |en, de    | au,sw    | tag1, tag2
2           | d2   |en, de    | au,sw    | tag1, tag2
3           | d3   |en, de    | au,sw    | tag2, tag3
Run Code Online (Sandbox Code Playgroud)

Ps 上面的表格只包含部分数据,实际查询返回更多的列,但上面的列在这里很重要。有人可以帮我解决这个问题。

Emi*_*zer 6

您可以使用GROUP BY具有string_agg这样的:

SELECT deliveries.deliver_id, deliver.name, 
       string_agg(distinct languages.display_name, ',' order by languages.display_name) as langs, 
       string_agg(distinct markets.default_name, ',' order by markets.default_name) as markets,
       string_agg(distinct tags.default_name, ',' order by tags.default_name) as tags,
       string_agg(distinct companies.name, ',' order by companies.name) as companies
    ...
    GROUP BY deliveries.deliver_id, deliver.name;
Run Code Online (Sandbox Code Playgroud)