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

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

ただいまの
回答率

90.12%

PDOで複文取得する方法が分からない

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 1,242

___knd

score 10

PHPでPDOを用いてmysqlからデータの取得を行おうとしております。
そこで、

$sql = 'SELECT * FROM ***;';
$stmt = $pdo -> query($sql);
if(!$stmt){
$info = $pfo ->errorInfo();
exit($info[2]);
}

while($data= $stmt ->fetch(PDO::FETCH_ASSOC)){
echo $data['**'];
}

このようにして必要なデータを取り出すことには成功したのですが、複文を書きたいときにエラーが出てしまいます。
具体的には

$sql = 'SELECT ** FROM ***;
        SELECT xx FROM xxx;';
$stmt = $pdo -> query($sql);
if(!$stmt){
$info = $pfo ->errorInfo();
exit($info[2]);
}

while($data= $stmt ->fetch(PDO::FETCH_ASSOC)){
echo $data['**'];
echo $data['xx'];

このように書いたのですがxxが見つからない?というようなエラーが出ます。

どのようにすれば複文記述することができるのかご教授願いたいです。説明不足であれば追記します。

環境はMAMPでPHPのバージョンは7.00です。

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


2016/1/10追記

ご解答ありがとうございます。

$sql = 'SELECT ** FROM ***;';
$sql1 = 'SELECT xx FROM xxx;';
$stmt = $pdo -> query($sql);
$stmt1 = $pdo -> query($sql1);

while($data= $stmt ->fetch(PDO::FETCH_ASSOC)){
echo $data['**'];
}
while($data1= $stmt1 ->fetch(PDO::FETCH_ASSOC)){
echo $data['xx'];
}

複文記述する際、このような感じでデータを取れることは理解しました。(命令をstmtに保存してるイメージなのかなという解釈です)
ただこれだと、[0],[1]…xx[0],xx[1]となってしまいます。

――――――――――――――
|[0]|xx[0]|
|
[1]|xx[1]|
|**[2]|xx[2]|
――――――――――――――
このように交互にデータの取得を行いたいです。

while文内では、データに命令されたデータを代入しある分を保持→表示しているのかなという解釈をしていて、
それだとwhile文の中にwhile文を書いてもやりたい処理はできなく困っています。

交互にデータを取ってくる処理のご教授をお願いしたいです。

何度もすみません。よろしくお願いします。

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

+1

対応策はg737a6bさんが記載してくださってるので、その解説でも…

PDOは基本的に複数のSQLを同時に発行出来ない仕様だったかと思います。
よくあるSQLインジェクション対策です。

SQL自体は正常に発行されて、
echo $data['xx']の行で$dataにはxxなんてキーは存在しねーよというエラーが出てるかと思います。
最初のSQL文章だけ処理されて、後者は切り捨てられてるからかと思います。
具体的な動作は確認してみてください。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/01/10 14:10

    解答ありがとうございます。

    そうですね…。***テーブルでmiyabiユーザーの情報(加入しているプランのID、年等)のコードがはいっていて、xxxテーブルでmiyabiユーザーが加入しているプランのIDを元にいくら費用が必要であるのか…というようなことをしたいです。

    今結合について少し調べてみましたが二つのテーブルを一つにしてそこからデータを抽出するということができるのですね。

    ちょっとやってみてるのですが…
    ***テーブルのプランIDとxxxテーブルのプランIDが一致した場合、***テーブルのプラン費用を取得したいです。
    $sql = 'SELECT ***.プラン費用
    FROM xxx INNER JOIN ***
    ON xxx.プランID = ***.プランID
    WHERE xxx.プランID = ***.プランID';

    こう書いてみているのですが、全てのデータが取得されてしまうのは何故なのでしょうか…

    初歩的な質問を重ねてしまってすみません。

    よろしくお願いします。

    キャンセル

  • 2016/01/10 18:03

    なるほどなるほど…
    やはりそのような意図があったのですね。

    >全てのデータが取得されてしまうのは何故なのでしょうか…
    結合条件と抽出条件が全く同じなのでフィルタリングされないからです。

    基本的にSQLの結合は直積です。
    更に高度になりますが集計という概念を使う感じになるかと思います。

    下記のように仮定してSQLを作ってみます
    ・xxxがユーザー一覧(ユーザーID, プランID)
    ・***がプラン一覧(プランID, プラン費用)

    SELECT
    xxx.ユーザーID,
    ***.プランID,
    ***.プラン費用
    FROM xxx
    INNER JOIN *** ON xxx.プランID = ***.プランID

    これで全ユーザーかつ全プランの情報が表示されます。
    例えば「今月の」「ユーザーIDが33番の」という条件をWhere句に入れてあげます

    WHERE date_format(xxx.加入日, '%Y-%m') = '2016-01'
    AND xxx.ユーザーID = 33

    もし、ユーザーが一度に複数プランに加入しつつ、合計金額を得たいのであればこんな感じになるでしょう。
    (usersテーブルを追加して考えます)
    これは集計関数を使ったさらに複雑な概念になりますが、分かれば強力な武器となるので頑張ってください

    SELECT
    users.id as user_id,
    sum(***.プラン費用) as amount
    FROM users
    INNER JOIN xxx ON xxx.ユーザーID = users.id
    INNER JOIN *** ON ***.プランID = xxx.プランID
    -- <- Where句で絞りたい場合は此処に差し込む
    GROUP BY users.id;

    キャンセル

  • 2016/01/10 21:52

    こちらの書いていない意図まで読み取っていただいてありがとうございます。

    >>全てのデータが取得されてしまうのは何故なのでしょうか…
    >結合条件と抽出条件が全く同じなのでフィルタリングされないからです。
    なるほど。そういわれるとそうですよね…超当たり前ですよね…。

    自分で調べて出来ました!!という報告をしたかったのですがもう少々お付き合いいただけると幸いですorz

    ・xxxがユーザー一覧(ユーザーID, プランカテゴリ、プランランク)
    ・***がプラン一覧(プランID, プラン費用)
    だとして、プランカテゴリがA,B,C…プランランクが001,002…プランIDがA001,A002…という状況です。
    なので、結合条件を***テーブルのプランIDとxxxテーブルのプランカテゴリ||xxxテーブルのプランランクとすれば
    ユーザーID|プランID|プランカテゴリ|プランランク|プラン費用
    たろう  | A001 | A | 001 | 100000
    はなこ | A002 | A | 002 | 150000
    といった結果が出てきてくれるのかなあと思ったのですが、
    たろう  | A001 | A | 001 | 100000
    たろう  | A001 | A | 002 | 150000
    はなこ  | A002 | A | 001 | 100000
    はなこ | A002 | A | 002 | 150000

    このようにすべての組み合わせが出てきてしまいます。

    今回、私のやりたいことは抽出条件と結合条件が一致しているので、WHERE文を書かずに終了するのではないのかなと考えているのですがいかがでしょうか。

    SELECT * FROM xxx INNER JOIN ***
    ON ***.プランID=xxx.プランカテゴリ||xxx.プランランク

    記述の仕方が悪いのでしょうか、考え方が間違えているのでしょうか…

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

    キャンセル

+1

命令文の数だけ$stmt = $pdo->query($sql);のようにステートメントを作成してください。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/01/10 12:27

    ありがとうございます。実行してみたところ、正直どのように処理されているのか分からなかったのですがなんとなくイメージできるようになりました。(理解はまだまだ浅いのですが…)

    しかし、やりたいことが上手くいかずに追記に書いたので、もしよろしければそちらも教えていただきたいです。よろしくお願いします。

    キャンセル

  • 2016/01/10 12:44

    交互に出力するのであれば、このようにすればどうですか。

    while( true ){
    $br = 0;

    if( $data = $stmt ->fetch(PDO::FETCH_ASSOC) ){
    echo $data["**"];
    }else{
    $br++;
    }

    if( $data1 = $stmt1 ->fetch(PDO::FETCH_ASSOC) ){
    echo $data["xx"];
    }else{
    $br++;
    }

    if( $br >= 2 ) break;
    }

    キャンセル

  • 2016/01/10 13:30

    解答ありがとうございます。

    なるほど。もし両方データがなくなったら終わるような仕組みを作ればいいのですね。私のやりたいことができました。ありがとうございました。

    キャンセル

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

  • ただいまの回答率 90.12%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる