繁体   English   中英

尝试按顺序时间戳时,ARRAY_AGG被GROUP BY中断

[英]ARRAY_AGG interrupted by GROUP BY when trying to ORDER BY timestamps

我为我的问题准备了一个SQL小提琴 -

鉴于下表:

CREATE TABLE chat(
    gid integer,            /* game id */
    uid integer,            /* user id */
    created timestamptz,
    msg text
);

填写以下测试数据:

INSERT INTO chat(gid, uid, created, msg) VALUES
    (10, 1, NOW() + interval '1 min', 'msg 1'),
    (10, 2, NOW() + interval '2 min', 'msg 2'),
    (10, 1, NOW() + interval '3 min', 'msg 3'),
    (10, 2, NOW() + interval '4 min', 'msg 4'),
    (10, 1, NOW() + interval '5 min', 'msg 5'),
    (10, 2, NOW() + interval '6 min', 'msg 6'),
    (20, 3, NOW() + interval '7 min', 'msg 7'),
    (20, 4, NOW() + interval '8 min', 'msg 8'),
    (20, 4, NOW() + interval '9 min', 'msg 9');

我可以使用以下查询:

SELECT json_object_agg(
          gid, array_to_json(y)
       ) FROM (
       SELECT  gid,
          array_agg(
            json_build_object(
              'uid', uid,
              'created', EXTRACT(EPOCH FROM created)::int,
              'msg', msg)
          ) y
        FROM  chat 
        GROUP BY gid /*, created
        ORDER BY created ASC */
) x;

使用“gid”作为键和数组值将表记录作为JSON对象获取:

{ "20" : [
    {"uid" : 3, "created" : 1514889490, "msg" : "msg 7"},
    {"uid" : 4, "created" : 1514889550, "msg" : "msg 8"},
    {"uid" : 4, "created" : 1514889610, "msg" : "msg 9"}
], 

"10" : [
    {"uid" : 1, "created" : 1514889130, "msg" : "msg 1"},
    {"uid" : 2, "created" : 1514889190, "msg" : "msg 2"},
    {"uid" : 1, "created" : 1514889250, "msg" : "msg 3"},
    {"uid" : 2, "created" : 1514889310, "msg" : "msg 4"},
    {"uid" : 1, "created" : 1514889370, "msg" : "msg 5"},
    {"uid" : 2, "created" : 1514889430, "msg" : "msg 6"}
] }

但是,我错过了一件小事,只是想不出来:

我需要通过“created”来命令数组。

所以我将ORDER BY created ASC添加到上面的查询中,并且还必须添加GROUP BY gid, created (您可以在上面的查询中看到有问题的代码注释掉)。

会破坏 array_agg并导致1元素数组(并且当从我的自定义存储函数返回为jsonb时,会覆盖重复的“gid”键):

{ "10": [{ "uid":2, "created":1514889800, "msg":"msg 6" }],

  "20": [{ "uid":4, "created":1514889980, "msg":"msg 9" }] }

这里甚至可以订购,还是应该在我的JAVA应用程序中使用它?

array_agg调用中指定ORDER BY

SELECT json_object_agg(gid, array_to_json(y))
FROM (
  SELECT  gid,
    array_agg(
      json_build_object(
        'uid', uid,
        'created', EXTRACT(EPOCH FROM created)::int,
        'msg', msg
      ) ORDER BY created -- Order the elements in the resulting array
    ) AS y
  FROM chat 
  GROUP BY gid
) x;

返回(使用jq格式化):

{
  "10": [
    {
      "uid": 1,
      "created": 1514890093,
      "msg": "msg 1"
    },
    {
      "uid": 2,
      "created": 1514890153,
      "msg": "msg 2"
    },
    {
      "uid": 1,
      "created": 1514890213,
      "msg": "msg 3"
    },
    {
      "uid": 2,
      "created": 1514890273,
      "msg": "msg 4"
    },
    {
      "uid": 1,
      "created": 1514890333,
      "msg": "msg 5"
    },
    {
      "uid": 2,
      "created": 1514890393,
      "msg": "msg 6"
    }
  ],
  "20": [
    {
      "uid": 3,
      "created": 1514890453,
      "msg": "msg 7"
    },
    {
      "uid": 4,
      "created": 1514890513,
      "msg": "msg 8"
    },
    {
      "uid": 4,
      "created": 1514890573,
      "msg": "msg 9"
    }
  ]
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM