「会社が仕事を担当する地域」を管理するDBを作っています。1社が複数の市区町村を担当する形態になっています。市区町村・都道府県で検索し、該当する会社を抽出したいです。
一番下にテストデータを記載してあります。ご回答よろしくお願い致します。
【テーブル構成】
[companyテーブル](id,company_name)
[cityテーブル](id,city_name,pref_id) ※cityテーブルはpref_idでprefテーブルと結びついています。
[prefテーブル](id.pref_name) ※prefは都道府県を意味しています。
[company_cityテーブル](id,company_id,city_id,number) ※companyとcityを結びつける中間テーブル
抽出したいデータ】
1.複数の市区町村ID(city_id)でAND検索し、該当するcompanyを抽出 例)city_idが1,2
2.複数の市区町村ID(city_id)でOR検索し、該当するcompanyを抽出 例)city_idが1,2
3.都道府県ID(pref_id)でAND検索し、該当するcompanyを抽出 例)pref_idが2,3
4.都道府県ID(pref_id)でOR検索し、該当するcompanyを抽出 例)pref_idが2,3
【出力形式】
-----------------------------------------------------------------
company_id, company_name, city_id, city_name, pref_id, pref_name
-----------------------------------------------------------------
company_idでDISTINCTした場合はどれか1つ、city_id,city_name,pref_id,pref_nameが取得できれば良い。できればcompnay_cityテーブルのnumberが1番のものが良いです。
上記形式で出力できない場合は company_id,company_nameだけの出力でも構いません。
=======================================
自分で実験してみたSQL
=======================================
以前こちら(https://oshiete.goo.ne.jp/qa/9171136.html)で教えて頂いたSQLを元に実験してみました。companyテーブルとcompany_cityテーブルを結びつけることはできますが、ここにcityテーブルとprefテーブルを結合する方法がよくわかりません。
【市区町村ID(city_id)でOR検索】
SELECT DISTINCT c.id, c.company_name, cc.city_id
FROM company AS c
INNER JOIN company_city AS cc ON c.id = cc.company_id
AND cc.city_id
IN ( 1, 2 )
WHERE 1
AND NOT (
cc.city_id IS NULL
)
[結果]DISTINCTしているのに重複データーが削除されてません。
id | company_name | city_id
1 | A社 | 1
1 | A社 | 2
2 | B社 | 2
【市区町村ID(city_id)でAND検索】
SELECT DISTINCT c.id, c.company_name
FROM company AS c
WHERE 1
and c.id in(
SELECT company_id FROM company_city as cc
WHERE city_id in(1,2)
GROUP BY company_id
HAVING COUNT(*)=2
)
[結果]こちらは求める結果になっていますが、SELECT DISTINCT c.id, c.company_name, cc.city_id のように city_idも表示させようとするとエラーになってしまいます。
id | company_name |
1 | A社 |
=======================================
テストデータ
=======================================
[cityテーブル]
CREATE TABLE IF NOT EXISTS `city` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`city_name` varchar(50) DEFAULT NULL,
`pref_id` int(2) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=5 ;
INSERT INTO `city` (`id`, `city_name`, `pref_id`) VALUES
(1, '青森市', 1),
(2, '弘前市', 1),
(3, '盛岡市', 2),
(4, '花巻市', 2);
[companyテーブル]
CREATE TABLE IF NOT EXISTS `company` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`company_name` varchar(50) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=4 ;
INSERT INTO `company` (`id`, `company_name`) VALUES
(1, 'A社'),
(2, 'B社'),
(3, 'C社');
[compnay_cityテーブル]
CREATE TABLE IF NOT EXISTS `company_city` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`company_id` int(11) NOT NULL,
`city_id` int(11) NOT NULL,
`number` int(2) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=7 ;
INSERT INTO `company_city` (`id`, `company_id`, `city_id`, `number`) VALUES
(1, 1, 1, 1),
(2, 1, 2, 2),
(3, 1, 3, 3),
(4, 2, 2, 2),
(5, 2, 3, 3),
(6, 3, 3, 3);
[pref]テーブル
CREATE TABLE IF NOT EXISTS `pref` (
`id` int(2) NOT NULL AUTO_INCREMENT,
`pref_name` varchar(50) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=4 ;
INSERT INTO `pref` (`id`, `pref_name`) VALUES
(1, '北海道'),
(2, '青森'),
(3, '岩手');
No.1ベストアンサー
- 回答日時:
順番に
1.city_idが1,2の両方をもつ
select company_id from company_city
where city_id in (1,2)
group by company_id
having count(distinct city_id)=2
//結果:1
2.city_idが1,2のいずれかもしくは両方をもつ
select distinct company_id from company_city
where city_id in (1,2)
//結果:1,2
3.pref_idが2,3の両方をもつ
select company_id from company_city as t1
inner join city as t2 on t1.city_id=t2.id
where t2.pref_id in (2,3)
group by t1.company_id
having count(distinct t2.pref_id)=2
//結果:なし
4.pref_idが2,3のいずれかもしくは両方をもつ
select distinct company_id from company_city as t1
inner join city as t2 on t1.city_id=t2.id
where t2.pref_id in (2,3)
//結果:1,2,3
yambejp様
ご回答ありがとうございます。
すべてのパターンで求める結果を得ることができました。
該当するcompany_idの抽出をシンプルに行い、取得できたcompany_idを使い、表示系は別途SQLを書いて対応してシステムを構築していきたいと思います。
今後ともよろしくお願いします。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- MySQL my_itemsテーブルのIDにAUTO_INCREMENT を追加ができるかで 1 2023/01/03 09:09
- MySQL エラー 1068 (42000): 複数の主キーが定義されていますエラー 2 2022/11/17 04:36
- JavaScript WordPressのコンタクトフォーム7にて送信者の位置情報を送らせたい 2 2022/09/14 23:28
- MySQL `picture` varchar(255) のコマンドで間違いないでしょうか? 1 2022/11/21 04:08
- MySQL テーブル作成です。どこかのスペルが間違っているか記号など スペースかな? 1 2022/10/01 05:08
- PHP php テーブルが作成できない 1 2022/11/17 23:41
- MySQL php テーブルを作れない 2 2022/11/17 18:22
- MySQL MYSQL エラー 2 2022/10/18 11:37
- MySQL PHPとMySQLを使った掲示板の作り方 1 2022/06/02 13:00
- PHP php エラー 2 2022/10/23 16:43
関連するカテゴリからQ&Aを探す
おすすめ情報
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
エクセルで最後の文字だけ置き...
-
SQLサーバから、項目の属性(型...
-
SQL Left Join で重複を排除す...
-
副問合せの書き方について
-
[MySQL] UNIQUE制約の値を更新...
-
select文のwhere句に配列を入れ...
-
SQLにて特定の文字を除いた検索...
-
selectした大量データをinsert...
-
エクセルの関数について教えて...
-
sqlで、600行あるテーブルを100...
-
MySQLのint型で001と表示する方...
-
VIEWの元のテーブルのindexって...
-
複数テーブルのGROUP BY の使い...
-
Unionした最後にGROUP BYを追加...
-
Access パラメータクエリをcsv...
-
LAST_INSERT_IDで同時にアクセ...
-
PL/SQLの変数について
-
[SQLServer] テーブル名からカ...
-
VMwareがCDドライブを認識する...
-
1対多結合で多を絞り込み条件と...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
エクセルで最後の文字だけ置き...
-
SQL Left Join で重複を排除す...
-
副問合せの書き方について
-
SQLサーバから、項目の属性(型...
-
VIEWの元のテーブルのindexって...
-
select文のwhere句に配列を入れ...
-
selectした大量データをinsert...
-
センノシド異性体構造式
-
Unionした最後にGROUP BYを追加...
-
insertを高速化させたい
-
SQLにて特定の文字を除いた検索...
-
マイクラPC版のコマンドで効率...
-
ある条件の最大値+1を初番する...
-
inner joinをすると数がおかし...
-
sqlで、600行あるテーブルを100...
-
エクセルの関数について教えて...
-
Access パラメータクエリをcsv...
-
URL と行番号の指定
-
複数テーブルのGROUP BY の使い...
-
PL/SQLの変数について
おすすめ情報