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

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

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

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

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

Q&A

解決済

4回答

2923閲覧

pg_query_params で where in を書く方法

morita4130002

総合スコア10

PostgreSQL

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

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

0グッド

0クリップ

投稿2020/01/09 06:32

環境

 PHP7.3 Postgres9.3

質問

 PHP プログラムで Postgres を操作しています。

 クエリの発行に pg_query_params() を使っているのですが、
この関数で、要素数が可変の where in はどのように書けばよいかがわかりません。
ご存じの方、ご教授いただけないでしょうか?

 例えば、次の SQL を実行する際、

select * from Users where id in (12345, 23456, 34567)  #id は int。

 PHP で次のように記述しますが、

pg_query_params( $con, 'select * from Users where id in ($1, $2, $3)', [12345, 23456, 34567] );

 where in の要素数が都度異なる場合、SQL 文字列内の ($1, $2, $3) の部分をどのように記述すればよいか、ということです。

 もちろん、要素数の分だけ <$インデックス, > をつなげた文字列を編集すればできるとは思いますが、
そうせずに記述する方法が無いだろうか、と考えています。

試したこと

 複数の要素をまとめた文字列をパラメータにしてみましたが、文字列として扱われてしまうため、エラーになってしまいました。

pg_query_params( $con, 'select * from Users where id in ($1)', ['12345, 23456, 34567'] );

以上、よろしくお願いいたします。

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

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

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

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

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

guest

回答4

0

地道に作りましょう。

php

1<?php 2$a = [123,456,789]; 3$r = range(1,count($a)); 4$params = []; 5foreach($r as $k=>$v){ 6 $params[] = '$'.$v; 7} 8echo implode(',',$params);

第三引数は$aそのまま突っ込む感じで。

投稿2020/01/09 06:48

m.ts10806

総合スコア80850

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

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

morita4130002

2020/01/09 06:56

やはりそうなのですね。ありがとうございました。
m.ts10806

2020/01/09 07:03 編集

なにに対して「やはりそうなのですね」なのでしょうか。 一応、可変対応したはずで、質問者さんが試してない内容なので「やはり」というのはおかしいように思いますが。
morita4130002

2020/01/09 07:42

お気を悪くされたのであればもうしわけありません。 要素数分の <$ インデックス, > を展開して SQL 文字列を組み立てればできることは はじめから想像できていたのですが、それをしない方法をお尋ねしたのです。
m.ts10806

2020/01/09 08:07

要素数が都度異なる場合 の"組み立て方"も思い付かれていないようでしたので、提示したまでです。
morita4130002

2020/01/09 08:26

どうもありがとうございました。勉強になりました。
guest

0

ベストアンサー

https://www.php.net/manual/ja/function.pg-query-params.php

マニュアルの「用例」に IN句の場合の書き方がそのまま載ってます。

これはだめ

php

1$valuelist = implode(', ', $values); 2$query = 'SELECT * FROM table1 WHERE col1 IN ($1)'; 3$result = pg_query_params($query, array($valuelist)) 4 or die(pg_last_error());

以下の二つは動くらしい。

php

1$valuelist = implode(', ', $values); 2$query = "SELECT * FROM table1 WHERE col1 IN ($valuelist)"; 3$result = pg_query($query) 4 or die(pg_last_error());

php

1$valuelist = '{' . implode(', ', $values . '}' 2$query = 'SELECT * FROM table1 WHERE col1 = ANY ($1)'; 3$result = pg_query_params($query, array($valuelist));

投稿2020/01/09 06:44

編集2020/01/09 06:52
退会済みユーザー

退会済みユーザー

総合スコア0

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

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

maisumakun

2020/01/09 06:46

これは失敗例で、「Produces error message: 'ERROR: invalid input syntax for integer'」と出てしまうと直後に書かれていました。
退会済みユーザー

退会済みユーザー

2020/01/09 06:50

あら、ほんとだ。
morita4130002

2020/01/09 07:03

ありがとうございます。後者の方を試してみます。
morita4130002

2020/01/09 07:32

情報ありがとうございました。やりたいことができました。 次のように any を使った書き方で動作しました。 pg_query_params($con, 'select * from Users where id = any ($1)', ['{' . implode(",", $ids) . '}']);
guest

0

もちろん、要素数の分だけ <$インデックス, > をつなげた文字列を編集すればできるとは思いますが、

pg_系の関数ではそれしかないかと思います。

PDOの場合はプレースホルダの入れ方が違って、?だけ書く、あるいは:nameのようにキーを指定する、という方法が使えます(PDO::prepare)。

投稿2020/01/09 06:40

maisumakun

総合スコア145183

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

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

morita4130002

2020/01/09 07:04

なるほど、そうなのですね。ありがとうございました!
guest

0

implode(",", [12345, 23456, 34567])

でどうでしょうか?

投稿2020/01/09 06:38

Masakin

総合スコア192

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

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

maisumakun

2020/01/09 06:41

それではプリペアした形に一致しないのではないでしょうか。
morita4130002

2020/01/09 07:07

ありがとうございます。 そうですね、implode() 結果が文字列になってしまうので、正しく動作しませんでした。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問