質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.36%
PostgreSQL

PostgreSQLはオープンソースのオブジェクトリレーショナルデータベース管理システムです。 Oracle Databaseで使われるPL/SQLを参考に実装されたビルトイン言語で、Windows、 Mac、Linux、UNIX、MSなどいくつものプラットフォームに対応しています。

Q&A

解決済

2回答

18618閲覧

ストアド内でテーブル変数のようなものを定義して、そこに値を追加していき、join等で使用する方法

hiepita1

総合スコア37

PostgreSQL

PostgreSQLはオープンソースのオブジェクトリレーショナルデータベース管理システムです。 Oracle Databaseで使われるPL/SQLを参考に実装されたビルトイン言語で、Windows、 Mac、Linux、UNIX、MSなどいくつものプラットフォームに対応しています。

0グッド

0クリップ

投稿2018/02/13 02:15

お世話になっております。
現在、SQLServerで書かれたストアドをpostgresqlのストアド(PL/pgSQL)に書き換える作業をしています。
その際に、SQLServerでいうところのテーブル変数がうまく書き換え出来ずに、詰まっています。
ご教示頂ければと思います。

やりたいこととしては、ストアド内でテーブル変数に値をpushしていき、場合によってはそのテーブル変数と他のテーブルをjoinするようなコードを書きたいです。
以下のようなイメージです。↓

CREATE FUNCTION public.test() RETURNS void AS $$ DECLARE -- sqlserverでいうテーブル変数のようなものを宣言 おそらくrecord型? recordhensu RECORD; -- 変数に追加する行の情報 gyou INTEGER; BEGIN -- レコード変数に行を追加していきたいが、方法がわからない。pushのようなメソッドがある? gyou := 1; rhensu.push(gyou); gyou := 2; rhensu.push(gyou); -- 他のテーブルと、レコード変数とをjoinしたものを取得 SELECT * FROM other_table INNER JOIN rhensu ON other_table.gyou = rhensu.gyou END; $$ LANGUAGE plpgsql;

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

tempにて作成してみました。テーブルで返します。

sql

1CREATE OR REPLACE FUNCTION public.test() RETURNS TABLE (Like hoge) 2AS $$ 3 DECLARE 4 gyou INTEGER; 5 6 BEGIN 7 -- レコード変数に行を追加していきたいが、方法がわからない。pushのようなメソッドがある? 8 create temp table if not exists rhensu (gyou int); 9 truncate rhensu; 10 gyou := 1; 11 insert into rhensu select gyou; 12 gyou := 2; 13 insert into rhensu select gyou; 14 15 -- 他のテーブルと、レコード変数とをjoinしたものを取得 16 RETURN QUERY SELECT hoge.* FROM hoge INNER JOIN rhensu ON hoge.gyou = rhensu.gyou; 17 END; 18$$ LANGUAGE plpgsql;

select * from test();

投稿2018/02/14 05:20

A.Ichi

総合スコア4070

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

hiepita1

2018/02/16 02:57

回答ありがとうございます。 テーブルを返すことも出来るんですね、参考にしたいと思います。
guest

0

ベストアンサー

忠実に再現するなら以下のようになります。
※因みにストアド内でselectする場合には、結果を受け取る記述が必要です。なので下記ではエラーになります。

SQL

1CREATE FUNCTION public.test() RETURNS void 2AS $$ 3 DECLARE 4 rhensu integer[]='{}'; 5 -- 変数に追加する行の情報 6 gyou INTEGER; 7 8 BEGIN 9 -- レコード変数に行を追加していきたいが、方法がわからない。pushのようなメソッドがある? 10 gyou := 1; 11 rhensu := rhensu || gyou ; 12 gyou := 2; 13 rhensu := rhensu || gyou ; 14 15 -- 他のテーブルと、レコード変数とをjoinしたものを取得 16 SELECT * FROM other_table INNER JOIN unnest(rhensu) as rhensu(gyou) ON other_table.gyou = rhensu.gyou; 17 END; 18$$ LANGUAGE plpgsql;

複数の項目を定義する場合はTYPE定義で構造体を作成し、その構造体を配列で定義します。
多分一時テーブルの方が利用目的には合うかもしれません。
record型は構造体へのポインタ変数に該当し、型自体は持ちません。

投稿2018/02/13 02:38

編集2018/02/13 04:05
sazi

総合スコア25302

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

hiepita1

2018/02/13 04:17

ありがとうございます。 record型はポインタ変数であるため、これに対してjoinは出来ないということですね。 一時テーブルを作成するようなイメージが近いので、本来であれば一時テーブルを作成してそれに対してレコードを追加していくような形で行いたいのですが、コード量が増える点と、SQLの問い合わせ回数が増えるので速度面で問題があるのではないか?と感じています。 (一時テーブルを利用する方法だと、Create temp table temp_table 型 ON COMMIT DROP; で作成して、それに対してINSERT INTO temp_table FROM gyou;のような書き方しか思い浮かばないのですが、もっと簡潔にかけますか?)
sazi

2018/02/16 03:30 編集

移植元のテーブル変数への代入は、select結果で代入していると思うので、 create temp table temp_table as select ~のような記述で良いかと思います。 そうでなければ、もう少し具体的な例示をお願いします。
hiepita1

2018/02/16 02:56

返事が遅れてしまって申し訳ありません。 そちらの方法でやってみたいと思います。ありがとうございます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問