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

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

ただいまの
回答率

87.78%

SQLでデータをうまく呼び出せない

解決済

回答 4

投稿 編集

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

score 403

画像の3行だけが表示されるはずなのですが、重複して2回取得してしまっています。

どこが原因でしょうか?

SQL文はこちらです。

SELECT UD.id,name5,name6,
US.id,US.user_id,skill,
UQ.id,UQ.user_id,quali

FROM user_data UD
LEFT JOIN user_skill US ON UD.id = US.user_id 
LEFT JOIN user_quali UQ ON UD.id = UQ.user_id

WHERE UD.id = 1;

イメージ説明


$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['技'];
    $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 />';
    }
}


↓ブラウザでの結果

キャラ名:サスケ

技名:千鳥
レベル:LV25

技名:天照
レベル:LV50

技名:月読
レベル:LV90

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


$en = filter_input(INPUT_GET, 'EN');
$sql = "SELECT UD.id,created,name5,name6,birth_year,birth_month,birth_day,gender,pref,city,
GROUP_CONCAT(skill),GROUP_CONCAT(US.year) AS skill_year,
GROUP_CONCAT(quali),GROUP_CONCAT(UQ.year) AS quali_year,GROUP_CONCAT(day),
lang,lang_deg,
major_category,major_skill,cont_date,introduction
FROM user_data UD
LEFT JOIN user_skill US ON UD.id = US.user_id
LEFT JOIN user_quali UQ ON UD.id = UQ.user_id
LEFT JOIN user_lang UL ON UD.id = UL.user_id
LEFT JOIN user_pr UP ON UD.id = UP.user_id
WHERE UD.id = '$en'
GROUP BY UD.id";
$stmt = $pdo->prepare($sql);
$stmt->execute();


var_dump($sql);


foreach ($stmt as $row) {
    echo '<br /><br />';
    echo 'EngineerNo.'.$row['id'].'<br />';
    echo '<img src="" alt="">';
    $birth = $row['birth_year'].sprintf('%02d', $row['birth_month']).sprintf('%02d', $row['birth_day']);
    echo '年齢:'.floor((date("Ymd")-$birth)/10000).'歳';
    echo ' | ';
    echo '所在:'.$row['pref'].' '.$row['city'];
    echo ' | ';
    echo '専攻カテゴリー:'.$row['major_category'];
    echo ' | ';
    echo '専攻スキル:'.$row['major_skill'];
    echo ' | ';
    echo '契約可能時期:'.$row['cont_date'].'<br /><br />';
    echo 'PR'.$row['introduction'].'<br /><br />';
    echo '<hr>';


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

    $level =$row['skill_year'];
    $levelArray = explode(",", $level);

    for($i = 0;$i < count($skillArray);$i++) {
        echo '<br />';
        echo 'スキル:'.$skillArray[$i].'<br />';
        echo '経験年数:'.$levelArray[$i].'<br />';
    }
}


echo '<br />';
var_dump($skill);
echo '<br />';
var_dump($skillArray);


↓ブラウザでの結果
EngineerNo.1
年齢:36歳 | 所在:東京都 台東区 | 専攻カテゴリー:システム開発/運用 | 専攻スキル:JAVA | 契約可能時期:1ヶ月以内

PR宜しくお願いします。

スキル:
経験年数:1年未満

NULL 
array(1) { [0]=> string(0) "" }


前者のナルトの例でいうと、string(20) "千鳥,天照,月読"を explodeに渡して配列作っているので
それと同じようにまず、
string() "JAVA,PHP,Javascript"を取得しなければならない。

実際、id=1の人の、user_dataテーブルに対する一番多いレコードは、user_skillテーブルに入っているJAVA,PHP,Javascriptの3レコードです。

user_qualiテーブルには2レコードが該当します。

なので画像のように取得したいのです。
イメージ説明

何回も同じものが表示されることはありません。
そのうえで、concatしています。

これと同じことも先ほどの後者のソースでもしないのです。
まずはconcatできる前の状態(重複して表示されていない状態)にしたいのです。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 4

checkベストアンサー

0

そういうときはクエリをいきなり構築するのではなく、ベースとなるテーブルに1つずつJOINするテーブルを増やして、原因となる部分を探し当てることからはじめましょう。今回の場合user_dataがベースになるので、user_skillだけをJOINして表示、次はuser_qualiをJOINして表示、と1つずつ試していってください。

 追記

1:nと1:nのテーブルをいっぺんにLEFT JOINさせればn件*n件のテーブルが表示されるのは「それが正しい動作」としか言えません。あなたは本来「どういうデータ」が欲しいのですか?

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/06/27 13:10 編集

    すいません。
    規模を小さくして原因調べているのですが、規模を一番小さくしてみても2回表示されてしまっています...

    質問を修正させていただきました。

    キャンセル

  • 2016/06/27 14:30

    すいません。
    こちらにて新しく質問させていただきました。
    https://teratail.com/questions/39286

    キャンセル

0

見たところすべてのテーブルのuser_idがすべて1なので全件取ってきているのでは。
上の3件だけ取るのであればUQ、UL、UPのidが1のものを取ってくるようにするべき?

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/06/27 18:05

    回答ありがとうございます。
    JOINではなくそれぞれのSQL文にすることで解決しそうです。

    キャンセル

0

テーブルレイアウトがわからないのでなんともですが、user_qualiの結合条件がuser_idだけだからだと思います。
スキルのID(カラム名はわかりません)も必要なのでは?

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/06/27 18:05

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

    キャンセル

0

skill が三つ、quali が二つで合計 6 行です。
これを三行にまとめると、quil は半端な数だけ表示されてしまいます。
どのように表示したらいいのですか?

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/06/27 18:05

    回答ありがとうございます。
    JOINではなくそれぞれのSQL文にすることで解決しそうです。

    キャンセル

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

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

関連した質問

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