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

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

ただいまの
回答率

88.06%

PDOで親子関係の配列を作りたいです

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 1,905
退会済みユーザー

退会済みユーザー

下記のような二つのテーブルをジョイントして 親子形式の配列に抽出したいと考えています。
イメージ説明

下記のような配列にしたいです。
data[parent][0][id]=> 1
data[parent][0][name]=> '白鳥 彩華'
data[parent][0][child][0][date]=> 2016-04-01
data[parent][0][child][0][weight]=> 54.3
data[parent][0][child][1][date]=> 2016-04-02
data[parent][0][child][1][weight]=> 57.6
data[parent][0][child][2][date]=> 2016-04-03
data[parent][0][child][2][weight]=> 55.8
以下省略...

「PDO::FETCH_NAMED」で出来そうなんですけど 使い方が解かりません
どなたか教えて下さい 宜しくお願い致します。

追記です...。
とりあえずの解決策として下記のコードで思った通りの配列は組めたんですけど...

$pdo = new PDO(DNS,DBUSER,DBPASS,array(PDO::ATTR_EMULATE_PREPARES => false));
$pdo->query('SET NAMES utf8');

$stmt = $pdo->query(" SELECT * FROM `parent` ");

if($stmt->rowCount() > 0){
    $articles = array();
    $i=0;
    while ($data = $stmt->fetch(PDO::FETCH_ASSOC)) {
        $articles[$i]['id'] = $data['parent_id'];
        $articles[$i]['name'] = $data['name'];
     $i++;
    }
    $i=0;
    foreach($articles as $parent){
        $id = $parent['id'];
        $stmt2 = $pdo->query(" SELECT * FROM `child` WHERE `parent_id` = '$id' ");
        while ($data2 = $stmt2->fetch(PDO::FETCH_ASSOC)) {
            $articles[$i]['child']['date'][] = $data2['date'];
            $articles[$i]['child']['weight'][] = $data2['weight'];
        }
     $i++;
    }
}
return $articles;


なんか 合理的ではない気がするんです (=ω=。)

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

+1

SQL でテーブルを結合するといいでしょうね。たとえば、こういうのはどうでしょう。

$sql = <<< SQL
SELECT
    `parent`.`parent_id` as `id`,
    `parent`.`name`,
    `child`.`date`,
    `child`.`weight`
  FROM `child` INNER JOIN `parent`
  ON `child`.`parent_id` = `parent`.`parent_id`;
SQL;

$id2index = array(); // id から $argicles での index を引くテーブル
$articles = array();
$stmt = $pdo->query($sql);
if ($stmt->rowCount() > 0) {
  while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
    if (array_key_exists($row['id'], $id2index)) { // すでにいる人の場合
      $index = $id2index[$row['id']];
    } else { // 新たに登場した人の場合
      $id2index[$row['id']] = $index = count($articles);
      $articles[] = array(
        'id' => $row['id'],
        'name' => $row['name'],
        'child' => array()
      );
    }
    $articles[$index]['child'][] = array(
      'date' => $row['date'],
      'weight' => $row['weight']
    );
  }
}

余計なことですが、次のようなコードにして添え字ではなく id や date をキーにした連想配列にしたほうが、よりフラットになってデータが扱いやすいような気がします。

$articles = array();
$stmt = $pdo->query($sql);
if ($stmt->rowCount() > 0) {
  while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
    if (! array_key_exists($row['id'], $articles)) {
      $articles[$row['id']] = array(
        'id' => $row['id'],
        'name' => $row['name'],
        'weight' => array()
      );
    }
    $articles[$row['id']]['weight'][$row['date']] = $row['weight'];
  }
}

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/04/17 23:38

    ありがとうございます><;;;
    こんなINNER JOINの使い方出来るんですね
    (同じテーブル内でしか使ったことありませんでした..(//∇//))
    それに変数をキーに配列を組むなんて思いもつきませんでした..

    まだまだ勉強しなきゃいけないことが いっぱいです
    今後とも どうそよろしくお願いいたしますm(_ _;)m

    キャンセル

  • 2016/04/18 06:16

    別テーブルの結合はごくごく普通ですよ、というか、RDBMS において正規化をしていくうえでの基本だと思います。
    あと、変数を配列のキーにするのもごくごく普通です。
    追記の部分のように、自分なりに考えて書いたコードがあれば、ただの丸投げでなく努力している様子がうかがえますし、コードから理解度や考え方などが見えてきてアドバイスもしやすくなると思います。

    キャンセル

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

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

関連した質問

同じタグがついた質問を見る