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

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

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

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

PHP

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

Q&A

解決済

1回答

1290閲覧

SQL THEN句の式にプレイスホルダが指定できない

aiueoao

総合スコア146

SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

PHP

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

0グッド

0クリップ

投稿2021/11/18 04:10

前提・実現したいこと

PHPでPostgreSQLの操作をしています.
|id|item|rank|count|
|:--|:--:|--:|
|1|hoge|1|0|
|2|hoge|2|0|
|3|fuga|1|0|

↓ 

|id|item|rank|count|
|:--|:--:|--:|
|1|hoge|1|10|
|2|hoge|2|20|
|3|fuga|1|0|

テーブルに対して複数行を一回でUpdateしたいです.
実行時に動的に任意のパラメータを渡したい&セキュリティの面から,プレイスホルダとbindValue()を使用しているのですが,下記コードだとエラーが出ます.
実行したい操作は,入力値「rank」の値とテーブルの同じ値の行を選択して,その行の「count」へ「rank」と同じIndexの入力値「count」を入れることです.

該当のソースコード

php

1<?php 2$db = new PDO("pgsql:host=localhost;dbname=postgres;user=postgres"); 3 4$data='{"rank":[1,2],"count":[10,20]}'; //入力値 5$obj=json_decode($data); 6 7$sql="UPDATE itemTable 8 SET count = CASE 9 WHEN rank=1 THEN :count0 10 WHEN rank=2 THEN :count1 11 ELSE Null 12 END 13 WHERE item=:item"; 14 15$stt=$db->prepare($sql); 16$stt->bindvalue(':item',"hoge"); 17 18$stt->bindvalue(':count0',$obj->count[0],PDO::PARAM_INT); 19$stt->bindvalue(':count1',$obj->count[1],PDO::PARAM_INT); 20 21$stt->execute(); //実行 22exit; 23

発生している問題・エラーメッセージ

Fatal error: Uncaught PDOException: SQLSTATE[42804]: Datatype mismatch: 7 ERROR: 列"count"は型integerですが、式は型textでした LINE 2: SET count = CASE ^ HINT: 式を書き換えるかキャストする必要があります。 in D:\Myfolder\XAMPP\htdocs\MyTest.php:23 Stack trace: #0 D:\Myfolder\XAMPP\htdocs\MyTest.php(23): PDOStatement->execute() #1 {main} thrown in D:\Myfolder\XAMPP\htdocs\MyTest.php on line 23

試したこと

プレイスホルダを使用せずに以下のコードで試したところ,エラーは出ずにUpdateされました.
このとき変更していないrank=2のほう(2つめのWHEN句)もUpdateできたのが余計に分かりません

php

1コード 2<?php 3$db = new PDO("pgsql:host=localhost;dbname=postgres;user=postgres"); 4 5$data='{"rank":[1,2],"count":[10,20]}'; //入力値 6$obj=json_decode($data); 7$countArr=$obj->count;           //追加 8 9$sql="UPDATE itemTable 10 SET count = CASE 11 WHEN rank=1 THEN $countArr[0] //変更 12 WHEN rank=2 THEN :count1 13 ELSE Null 14 END 15 WHERE item=:item"; 16 17$stt=$db->prepare($sql); 18$stt->bindvalue(':item',"hoge"); 19 20//$stt->bindvalue(':count0',$obj->count[0],PDO::PARAM_INT); コメントアウト 21$stt->bindvalue(':count1',$obj->count[1],PDO::PARAM_INT); 22 23$stt->execute(); //実行 24exit;

補足情報(FW/ツールのバージョンなど)

Windows10
PostgreSQL 12.8
PHP 7.2.18

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

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

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

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

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

yambejp

2021/11/18 04:53

var_dump($obj->count[0])で得られる値はなんでしょうか?
aiueoao

2021/11/18 06:10

int(10) となりました
guest

回答1

0

ベストアンサー

プレスホルダーが展開された時に、文字として展開されることになるので、展開されたものをCASTする必要があるんだと思います。

SET count = CASE WHEN rank=1 THEN cast(:count0 as integer) WHEN rank=2 THEN cast(:count1 as integer) ELSE Null END

投稿2021/11/18 06:58

sazi

総合スコア25327

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

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

aiueoao

2021/11/18 07:04

CASTしたら出来ました! 回答ありがとうございました!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問