106

row_to_json()PostgreSQL 9.2 で追加された関数を使用して、クエリの結果を JSON にマップしようとしています。

結合された行をネストされたオブジェクト (1:1 の関係) として表現する最良の方法がわかりません。

これが私が試したことです(セットアップコード:テーブル、サンプルデータ、その後にクエリ):

-- some test tables to start out with:
create table role_duties (
    id serial primary key,
    name varchar
);

create table user_roles (
    id serial primary key,
    name varchar,
    description varchar,
    duty_id int, foreign key (duty_id) references role_duties(id)
);

create table users (
    id serial primary key,
    name varchar,
    email varchar,
    user_role_id int, foreign key (user_role_id) references user_roles(id)
);

DO $$
DECLARE duty_id int;
DECLARE role_id int;
begin
insert into role_duties (name) values ('Script Execution') returning id into duty_id;
insert into user_roles (name, description, duty_id) values ('admin', 'Administrative duties in the system', duty_id) returning id into role_id;
insert into users (name, email, user_role_id) values ('Dan', 'someemail@gmail.com', role_id);
END$$;

クエリ自体:

select row_to_json(row)
from (
    select u.*, ROW(ur.*::user_roles, ROW(d.*::role_duties)) as user_role 
    from users u
    inner join user_roles ur on ur.id = u.user_role_id
    inner join role_duties d on d.id = ur.duty_id
) row;

を使用ROW()すると、結果のフィールドを子オブジェクトに分離できることがわかりましたが、単一のレベルに制限されているようです。AS XXXこの場合は必要だと思うので、これ以上ステートメントを挿入することはできません。

::user_rolesそのテーブルの結果の場合、たとえば を使用して、適切なレコード型にキャストするため、列名が与えられます。

そのクエリが返すものは次のとおりです。

{
   "id":1,
   "name":"Dan",
   "email":"someemail@gmail.com",
   "user_role_id":1,
   "user_role":{
      "f1":{
         "id":1,
         "name":"admin",
         "description":"Administrative duties in the system",
         "duty_id":1
      },
      "f2":{
         "f1":{
            "id":1,
            "name":"Script Execution"
         }
      }
   }
}

私がやりたいことは、結合を追加できる方法で結合用の JSON を生成し (ここでも 1:1 で問題ありません)、結合先の親の子オブジェクトとして表すことができます。つまり、次のようになります。

{
   "id":1,
   "name":"Dan",
   "email":"someemail@gmail.com",
   "user_role_id":1,
   "user_role":{
         "id":1,
         "name":"admin",
         "description":"Administrative duties in the system",
         "duty_id":1
         "duty":{
            "id":1,
            "name":"Script Execution"
         }
      }
   }
}
4

3 に答える 3

201

更新: PostgreSQL 9.4では、 、およびの導入によりto_jsonjson_build_objectjson_objectjson_build_arrayこれが大幅に改善されましたが、すべてのフィールドに明示的に名前を付ける必要があるため冗長になっています。

select
        json_build_object(
                'id', u.id,
                'name', u.name,
                'email', u.email,
                'user_role_id', u.user_role_id,
                'user_role', json_build_object(
                        'id', ur.id,
                        'name', ur.name,
                        'description', ur.description,
                        'duty_id', ur.duty_id,
                        'duty', json_build_object(
                                'id', d.id,
                                'name', d.name
                        )
                )
    )
from users u
inner join user_roles ur on ur.id = u.user_role_id
inner join role_duties d on d.id = ur.duty_id;

古いバージョンについては、読み進めてください。


一列に限らず、ちょっと辛いだけです。を使用して複合行型にエイリアスを設定することはできないASため、効果を得るにはエイリアス化されたサブクエリ式または CTE を使用する必要があります。

select row_to_json(row)
from (
    select u.*, urd AS user_role
    from users u
    inner join (
        select ur.*, d
        from user_roles ur
        inner join role_duties d on d.id = ur.duty_id
    ) urd(id,name,description,duty_id,duty) on urd.id = u.user_role_id
) row;

http://jsonprettyprint.com/経由で生成されます:

{
  "id": 1,
  "name": "Dan",
  "email": "someemail@gmail.com",
  "user_role_id": 1,
  "user_role": {
    "id": 1,
    "name": "admin",
    "description": "Administrative duties in the system",
    "duty_id": 1,
    "duty": {
      "id": 1,
      "name": "Script Execution"
    }
  }
}

array_to_json(array_agg(...))ところで、1対多の関係があるときに使用したいと思うでしょう。

上記のクエリは、理想的には次のように記述できる必要があります。

select row_to_json(
    ROW(u.*, ROW(ur.*, d AS duty) AS user_role)
)
from users u
inner join user_roles ur on ur.id = u.user_role_id
inner join role_duties d on d.id = ur.duty_id;

...しかし、PostgreSQL のコンストラクターは列のエイリアスROWを受け入れません。AS悲しいことに。

ありがたいことに、彼らは同じように最適化します。プランを比較します。

CTE は最適化フェンスであるため、ネストされたサブクエリのバージョンを言い換えてチェーンされた CTE (WITH式) を使用すると、パフォーマンスが低下する可能性があり、同じ計画にはなりません。この場合、コンストラクターrow_to_jsonの列名をより直接的にオーバーライドするか、いくつかの改善が得られるまで、醜いネストされたサブクエリに悩まされます。ROW


とにかく、一般的に、原則は、列を持つjsonオブジェクトを作成したい場所でありa, b, c、違法な構文を書くことができればいいのにと思います:

ROW(a, b, c) AS outername(name1, name2, name3)

代わりに、行型の値を返すスカラー サブクエリを使用できます。

(SELECT x FROM (SELECT a AS name1, b AS name2, c AS name3) x) AS outername

または:

(SELECT x FROM (SELECT a, b, c) AS x(name1, name2, name3)) AS outername

さらに、追加の引用符なしで値を構成できることに注意してくださいjson。たとえば、 a の出力を ajson_agg内に配置するrow_to_jsonと、内部のjson_agg結果は文字列として引用されず、json として直接組み込まれます。

たとえば、任意の例では:

SELECT row_to_json(
        (SELECT x FROM (SELECT
                1 AS k1,
                2 AS k2,
                (SELECT json_agg( (SELECT x FROM (SELECT 1 AS a, 2 AS b) x) )
                 FROM generate_series(1,2) ) AS k3
        ) x),
        true
);

出力は次のとおりです。

{"k1":1,
 "k2":2,
 "k3":[{"a":1,"b":2}, 
 {"a":1,"b":2}]}

json_agg[{"a":1,"b":2}, {"a":1,"b":2}]が再びエスケープされていないことに注意してくださいtext

これは、json 操作を構成して行を構築できることを意味します。非常に複雑な PostgreSQL 複合型を常に作成row_to_jsonして、出力を呼び出す必要はありません。

于 2012-11-05T07:03:06.353 に答える
2

長期にわたる保守性のための私の提案は、VIEW を使用してクエリの大まかなバージョンを作成し、次に以下の関数を使用することです。

CREATE OR REPLACE FUNCTION fnc_query_prominence_users( )
RETURNS json AS $$
DECLARE
    d_result            json;
BEGIN
    SELECT      ARRAY_TO_JSON(
                    ARRAY_AGG(
                        ROW_TO_JSON(
                            CAST(ROW(users.*) AS prominence.users)
                        )
                    )
                )
        INTO    d_result
        FROM    prominence.users;
    RETURN d_result;
END; $$
LANGUAGE plpgsql
SECURITY INVOKER;

この場合、オブジェクト prominence.users はビューです。users.* を選択したので、ビューを更新してユーザー レコードにより多くのフィールドを含める必要がある場合でも、この関数を更新する必要はありません。

于 2016-02-10T23:32:08.010 に答える