pgsql 1到n关系到json

时间:2014-05-08 07:36:10

标签: sql json postgresql left-join hstore

长话短说,如何使用1到n选择数据来构建json,如示例所示:

SELECT table1.id AS id1,table2.id AS id2,t_id,label 
FROM table1 LEFT JOIN table2 ON table2.t_id = table1.id 

result 
|id1|id2|t_id|label|
+---+---+----+-----+
|1  | 1 | 1  | a   |
|   | 2 | 1  | b   |
|   | 3 | 1  | c   |
|   | 4 | 1  | d   |
|2  | 5 | 2  | x   |
|   | 6 | 2  | y   |

变成这个

 SELECT table1.id, build_json(table2.id,table2.label) AS json_data 
 FROM table1 JOIN table2 ON table2.t_id = table1.id  
 GROUP BY table1.id

|id1|json_data
+--+-----------------
|1 |{"1":"a","2":"b","3":"c","4":"d"}
|2 |{"5":"x","6":"y"}

我猜最好的开始是从列

构建一个数组

Hstore代替json也可以

1 个答案:

答案 0 :(得分:0)

你的表结构有点奇怪(看起来更像是报表而不是表),所以我在这里看到两个任务:

使用正确的id1替换空值。你可以这样做

with cte1 as (
    select
        sum(case when id1 is null then 0 else 1 end) over (order by t_id) as id1_partition,
        id1, id2, label
    from Table1  
), cte2 as (
    select
        first_value(id1) over(partition by id1_partition) as id1,
        id2, label
    from cte1
)
select *
from cte2

现在您必须汇总数据json。据我所知,PostgreSQL中没有这样的功能,所以你必须手动连接数据:

with cte1 as (
    select
        sum(case when id1 is null then 0 else 1 end) over (order by t_id) as id1_partition,
        id1, id2, label
    from Table1  
), cte2 as (
    select
        first_value(id1) over(partition by id1_partition) as id1,
        id2, label
    from cte1
)
select
   id1,
   ('{' || string_agg('"' || id2 || '":' || to_json(label), ',') || '}')::json as json_data
from cte2
group by id1

<强> sql fiddle demo

如果您想转换为hstore

with cte1 as (
    select
        sum(case when id1 is null then 0 else 1 end) over (order by t_id) as id1_partition,
        id1, id2, label
    from Table1  
), cte2 as (
    select
        first_value(id1) over(partition by id1_partition) as id1,
        id2, label
    from cte1
)
select
    c.id1, hstore(array_agg(c.id2)::text[], array_agg(c.label)::text[])
from cte2 as c
group by c.id1

<强> sql fiddle demo