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

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

新規登録して質問してみよう
ただいま回答率
85.34%
PostgreSQL

PostgreSQLはオープンソースのオブジェクトリレーショナルデータベース管理システムです。 Oracle Databaseで使われるPL/SQLを参考に実装されたビルトイン言語で、Windows、 Mac、Linux、UNIX、MSなどいくつものプラットフォームに対応しています。

SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

Q&A

解決済

2回答

1441閲覧

3つのテーブルを結合し、重複があった場合IDの大きいものを一件抽出する場合のSQLについて

yowayowanitohei

総合スコア31

PostgreSQL

PostgreSQLはオープンソースのオブジェクトリレーショナルデータベース管理システムです。 Oracle Databaseで使われるPL/SQLを参考に実装されたビルトイン言語で、Windows、 Mac、Linux、UNIX、MSなどいくつものプラットフォームに対応しています。

SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

0グッド

0クリップ

投稿2021/04/19 04:58

編集2021/04/19 08:18

3つのテーブルを結合し、結果を取得しようとしておりますが、
該当結果が複数あった場合にIDが大きいものを
一件だけ抽出する処理の書き方がわからず困っております。

3つのテーブルの用意があります。

上記3つのテーブルを結合し、店コードと日付で検索を行い結果を取得したいです。

街テーブルからは緯度経度をもとに計算を行い結果をもとに結合を行っておりますが
緯度経度の性質上、隣り合った市区町村で緯度経度が被ってしまうことがあり、
もし緯度経度で複数の市区町村が抽出された場合は街コードが大きいものを取得したいと考えております。


試したこと
以下のsqlで店コードが指定のものかつ日付が指定の範囲でいちばん大きい結果を取得
することができましたが、meshcodeが複数該当してしまった場合に街コードが大きいものを一件
取り出す抽出方法がわかりませんでした。。。

SELECT stores.*,categories.category,cities.cityname FROM(SELECT storecode,buildcode,lat.lng,date,rank() OVER (PARTITION BY storecode ORDER BY date DESC) AS rank FROM stores          WHERE (storecode = 'A01' AND date <= '2021-04-19') OR (storecode = 'B02' AND date <= '2021-04-20') AS storetable LEFT OUTER JOIN categories ON stores.buildcode = categories.buildcode LEFT OUTER JOIN meshcode ON (latとlngを元に計算する式の為省略) = cities.meshcode WHERE rank = 1

複数のテーブルを結合しつつ同時に複数条件で検索を行う際に
それぞれの該当結果が複数ある場合での結果をそれぞれ一件に絞り込む方法について
ご存知でしたらお伺いしたいです。

質問が長く見づらくなってしまい大変お手数をおかけしますが、どうぞよろしくお願い致します。
また、わかりにくい点や説明不足/不明点等ございましたらご指摘をお願い致します。

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

sazi

2021/04/19 08:04 編集

質問のSQLとテーブルが合っていません。 SQLにあるmeshcodeはcitiesの事ですか? 結合条件が省略されていますが、storesとの結合ですよね。
sazi

2021/04/19 06:46 編集

> 以下のsqlで店コードが指定のものかつ日付が指定の範囲でいちばん大きい結果を取得 することができました 「日付が指定の範囲でいちばん大きい結果」を求めている部分はどこですか? rank()で順番は作り出していますが、限定する部分は見当たりませんが。
sazi

2021/04/19 06:48

直接関係はないですけど、where条件は間違っていませんか? WHERE storecode = 'A01' AND date <= '2021-04-19' OR storecode = 'B02' AND date <= '2021-04-20' ではなくて WHERE (storecode = 'A01' AND date <= '2021-04-19') OR (storecode = 'B02' AND date <= '2021-04-20') だと思いますけど。
yowayowanitohei

2021/04/19 07:50

sazi様 いつもご回答ありがとうございます。 失礼しました。。。最後の一文が抜けておりました。。。 条件に関してもご指摘の通りです。 where rank = 1 質問文修正させていただきます。
sazi

2021/04/19 08:00

最初のコメントにも対応して下さい。
guest

回答2

0

やりたいことは以下であっていますか?

  • storesを何らかの条件で検索する。ただし、同じstore_idでdateが異なる行があった場合は、dateがより新しい行だけが欲しい。
  • categoriesとcitiesのデータも一緒に欲しいが、citiesについては、meshcodeが同じ場合は、citycodeがより小さい行と紐づけたい

あっている場合、方法はいろいろあると思いますが、たとえば以下のようなクエリで実現できるかと思います。

SQL

1SELECT * 2FROM ( 3 SELECT *, RANK() OVER (PARTITION BY storecode ORDER BY date DESC) AS rnk 4 FROM stores 5 WHERE (storecode = 'A01' AND date <= '2021-04-19') 6 OR (storecode = 'B02' AND date <= '2021-04-20') 7) AS s 8LEFT OUTER JOIN categories ON s.buildcode = categories.buildcode 9LEFT OUTER JOIN ( 10 SELECT *, RANK() OVER (PARTITION BY meshcode ORDER BY citycode) AS rnk 11 FROM cities 12) AS t ON (latとlngを元に計算する式の為省略) = t.meshcode 13WHERE s.rnk = 1 AND t.rnk = 1

投稿2021/04/19 11:58

neko_the_shadow

総合スコア2351

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

yowayowanitohei

2021/04/19 13:56

neko_the_shadow様 いつもご回答ありがとうございます! 上記コードにて動作できました。。。! SQLちゃんと書ける前にまず質問しっかりできるようになります(泣) 拙い質問からご忖度ありがとうございました。非常に助かりました
guest

0

ベストアンサー

質問内容を忖度すると、こんな感じかと思います。

SQL

1with target_stores as ( 2 SELECT * 3 FROM stores 4 WHERE (storecode = 'A01' AND date <= '2021-04-19') 5 OR (storecode = 'B02' AND date <= '2021-04-20') 6) 7select 8from target_stores 9 left join categories 10 on target_stores.buildcode = categories.buildcode 11 left join meshcode 12 on (latとlngを元に計算する式の為省略) = meshcodes.meshcode 13where not exists( 14 select 1 from target_stores max_date 15 where max_date.storecode=target_stores.storecode 16 and max_date.date > target_stores.date 17 ) 18 and not exists( 19 select 1 from meshcode max_citycode 20 where max_citycode.meshcode=meshcode.meshcode 21 and max_citycode.citycode > meshcode.citycode 22 )

投稿2021/04/19 07:05

sazi

総合スコア25331

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

yowayowanitohei

2021/04/19 07:55

sazi様 わかりやすいご回答ありがとうございます。 追加で何度もご質問になってしまい大変申し訳ないのですが max_dateとmax_citycodeについて意図した動作がよくわからず 確認させていただきたいです。。。 これはこのクエリとは別に設定が必要との認識でよろしいでしょうか。
sazi

2021/04/19 08:01

> max_dateとmax_citycodeについて意図した動作がよくわからず max_dateとmax_citycodeは相関副問合せで問合せするテーブル名が同じなので、別名を付けています。
sazi

2021/04/19 08:03 編集

相関副問合せで揃えましたが、ネストさせれば、meshcodeもrank()を使用すれば同じように出来ますね。
yowayowanitohei

2021/04/19 13:58 編集

理解に時間かかりましたが、動作できました! 質問抜け漏れないよう、しっかり記入するようにします泣 今後ともどうぞよろしくお願いします。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.34%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問