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

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

ただいまの
回答率

87.59%

1:n構造のnテーブルから1つ1つのカラムを取得したい

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 1,165

score 403

1:n構造のnテーブルに複数のレコードが入ってます。

まず、同じ表示が2回されていることが問題でしたが、GROUP BYで1つにまとめることで2回表示されることはなくなりました。

イメージ説明


GROUP_CONCATで、nテーブルに入っている複数のレコードをまとめることができました。
イメージ説明

$en = filter_input(INPUT_GET, 'EN');
$sql = "SELECT name.id, name AS 'キャラ名', user_id,GROUP_CONCAT(skill) AS '技',GROUP_CONCAT(level) AS 'レベル' FROM name LEFT JOIN skill ON name.id = skill.user_id WHERE name.id = '$en' GROUP BY name.id";
$stmt = $pdo->prepare($sql);
$stmt->execute();

var_dump($sql);

foreach ($stmt as $row) {
    echo '<br /><br />';
    echo 'キャラ名:'.$row['キャラ名'].'<br />';
    echo '技名:'.$row['技'].'<br />';
    echo 'レベル:'.$row['レベル'].'<br />';
}

でも、実際には、こういう風に分けて表示したい。

イメージ説明

そこで思いついたのがこのような方法でした。

$en = filter_input(INPUT_GET, 'EN');
$sql = "SELECT name.id, name AS 'キャラ名', user_id,GROUP_CONCAT(skill) AS '技',GROUP_CONCAT(level) AS 'レベル' FROM name LEFT JOIN skill ON name.id = skill.user_id WHERE name.id = '$en' GROUP BY name.id";
$stmt = $pdo->prepare($sql);
$stmt->execute();

var_dump($sql);

foreach ($stmt as $row) {
    echo '<br /><br />';
    echo 'キャラ名:'.$row['キャラ名'].'<br />';

    $skill =$row['技'];
    $skill1 = mb_substr($skill, 0, 3);
    $skill2 = mb_substr($skill, 4, 5);

    $level =$row['レベル'];
    $level1 = mb_substr($level, 0, 4);
    $level2 = mb_substr($level, 5, 4);
    echo '<br />';
    echo '技名:'.$skill1.'<br />';
    echo 'レベル:'.$level1.'<br />';
    echo '<br />';
    echo '技名:'.$skill2.'<br />';
    echo 'レベル:'.$level2.'<br />';
}

上記の記述だとスキルが増えたときに対応できないし、
スキル名によっても切り取り位置が変わる。

なにか良い方法はありませんでしょうか?

ナルトの例でいえば、
キャラ名は1回表示されるだけでいいからGROUP BYは必要そう。
技とレベルはあるだけ表示させたいから、GROUP_CONCATを使う?
でもそれすると、纏めて取るから、技とレベルを個別に取り出せなくなる。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

checkベストアンサー

0

表示の構成的に、名前と技は別クエリで取るべきではないでしょうか。
こういう形の表示をしたいのでしたら、そもそもGROUP BYするのではなく、nameのリストをforeachで回し、foreach内で該当idのskillをリストで取る、と2段階のクエリにするべきかと思います。

クエリを変える気が無いのでしたら、GROUP_CONCATで繋げたカラムを,区切りで再度分割するのが手っ取り早いのではないでしょうか。

foreach ($stmt as $row) {
    echo '<br /><br />';
    echo 'キャラ名:'.$row['キャラ名'].'<br />';

    $skill =$row['技'];
    $skillArray = explode(",", $skill);

    $level =$row['レベル'];
    $levelArray = explode(",", $level);

    for($i = 0;$i < count($skillArray);$i++) {
        echo '<br />';
        echo '技名:'.$skillArray[$i].'<br />';
        echo 'レベル:'.$levelArray[$i].'<br />';
    }
}

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/06/27 12:30

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

    explode( 分解文字, 文字列, [分解数] )

    キャラ名:サスケ

    技名:千鳥
    レベル:LV25

    技名:天照
    レベル:LV50

    技名:月読
    レベル:LV90

    string(20) "千鳥,天照,月読"
    array(3) { [0]=> string(6) "千鳥" [1]=> string(6) "天照" [2]=> string(6) "月読" }

    とりあえず
    explode(",", カンマで区切られた文字列)
    explode(",", $level)

    にすればいいことは分かりましたが
    第一引数と第二引数は、例えばどういう場面でどう使ったりするのでしょうか?
    マニュアルは勿論見ています。

    キャンセル

  • 2016/06/27 13:04

    explode( 分解文字, 文字列, [分解数] )
    ここにそのまま書いてあるとおりです。
    いまあなたのクエリで得られた「技」「レベル」はカンマ区切りのデータですので、分解文字にはカンマを使い、分解したい文字列は「技」や「レベル」の結果ということになります。タブ文字区切りなら分解文字はタブ文字になりますし、&で句切られていいるなら分解文字は&になります。

    キャンセル

  • 2016/06/27 13:15

    なんか勘違いしていたかもです。

    explode(",", $skill);
    ↑第3引数に$skillがあるように見えたのですが、
    第1引数に","
    第2引数に $skillがあるということですね...

    explode( 分解文字, 文字列, [分解数] )
    なのでなんで分解数に$skillがあるんだろうと思ってました汗

    キャンセル

0

select name,group_concat(concat(`skill_name`,',',`level`) separator ';') as skill


のように、セパレータを工夫して複数取り出すか、
SQL文を2つにわけるかのどちらかでしょう

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/06/27 14:29

    ありがとうございます!

    キャンセル

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

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

関連した質問

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