▼tb_job(仕事)
|job_id|name|
|1|漁師|
|2|医者|
|3|農家|
|4|警官|
▼tb_contact(申し込み)
|contact_id|job_id|user|
|1|2|佐藤|
|2|2|田中|
|3|1|本村|
各仕事の申込数を以下のように出力したいと思っています。
▼理想
|job_id|name|申込数|
|1|漁師|1|
|2|医者|2|
|3|農家|0|
|4|警官|0|
そこで上記2つ「tb_contact」を「job_id」でGROUP BYして
SQLで以下のように結合するのですが
SELECT *,COUNT(*) FROM tb_contact RIGHT JOIN tb_job ON tb_contact.job_id = tb_job.job_id GROUP BY tb_contact.job_id;
下記のように申し込みがある仕事については集計できるのですが、
申し込みがない仕事については1括りにされてしまい、COUNTにも
その合計が表示されてしまいます。
▼結果
|contact_id|job_id|user|job_id|name|COUNT(*)
|NULL|NULL|NULL|3|農家|2
|3|1|本村|1|漁師|1
|1|2|佐藤|2|医者|2
「▼理想」のように出力するにはどの様なSQL文を投げれば良いのでしょうか。
ご教授のほど宜しくお願い致します。
↓
コレに対する回答
とりあえず、一番の原因はgroup by に 「tb_contactの」job_idを指定していることです。
tb_contactのjob_idに、3や4の値はないわけで、right joinした結果も、その値は当然nullです。
その結果「group by の値はnull」として、一列になってしまいます。
ということで、元のSQL文への変更を最小限にするなら
tb_contactのjob_idに、3や4の値はないわけで、right joinした結果も、その値は当然nullです。
その結果「group by の値はnull」として、一列になってしまいます。
ということで、元のSQL文への変更を最小限にするなら
となります。
補足として、
group by または集計関数 を使用する時、Selectに「group by に指定されていない集計関数以外の列」を指定できてしまうのはMYSQLの独自拡張です。
今回も最初のSQLが「通ってしまう」ことが誤解の一員になっているんじゃないかと思います。
参考:寛容なMySQLを非寛容にすること(その3) | inquisitor
それをふまえた上で「理想」と同じ出力にするなら、こんな感じで。
補足として、
group by または集計関数 を使用する時、Selectに「group by に指定されていない集計関数以外の列」を指定できてしまうのはMYSQLの独自拡張です。
今回も最初のSQLが「通ってしまう」ことが誤解の一員になっているんじゃないかと思います。
参考:寛容なMySQLを非寛容にすること(その3) | inquisitor
それをふまえた上で「理想」と同じ出力にするなら、こんな感じで。
ちなみに、私の場合、
- JOINするテーブルには別名をつける(同じテーブルを複数JOINすることも多々あるため)
- RIGHT JOINは使用しない(RIGHT JOINはLEFT JOINに置き換え可能で、統一したほうが可読性が良くなるため)
というのを習慣にしているので、私が書くならこんな感じになります。
こちらのページをそのままメモ