2012-05-15 1 views
11

Angenommen, ein Szenario ähnlich zu diesem question. Ich mag die folgenden Ergebnisse erhalten:sql - Gruppe von in Bereichen, um Bereiche ohne Werte zu enthalten

score range | number of occurrences 
------------------------------------- 
    0-9  |  11 
    10-19  |  14 
    20-29  |   3 
    ...  |  ... 

Und ich kann die ausgewählte Antwort als Lösung verwenden:

select t.range as [score range], count(*) as [number of occurences] 
from (
    select case 
    when score between 0 and 9 then ' 0- 9' 
    when score between 10 and 19 then '10-19' 
    else '20-99' end as range 
    from scores) t 
group by t.range 

Wie kann ich versichern, dass der Score-Bereich von 30 bis 39 wird Anzeige, selbst wenn Es gibt keine Ergebnisse in diesem Bereich?

+0

nicht wirklich eine Antwort auf Ihre Frage, aber warum nicht tun Sie es einfach auf der Codierungsseite? –

+0

Eine Dummy-Range-Tabelle, oder gehen Sie mit Ben Lees Ratschlag ... – Wrikken

+0

@BenLee Das ist, was ich gerade mache. aber ich mag es, die ganze Logik zusammen zu halten – dcarneiro

Antwort

18

Diese Abfrage (auch auf SQL Fiddle):

WITH ranges AS (
    SELECT (ten*10)::text||'-'||(ten*10+9)::text AS range, 
      ten*10 AS r_min, ten*10+9 AS r_max 
     FROM generate_series(0,9) AS t(ten)) 
SELECT r.range, count(s.*) 
    FROM ranges r 
    LEFT JOIN scores s ON s.score BETWEEN r.r_min AND r.r_max 
GROUP BY r.range 
ORDER BY r.range; 

EDIT:

Sie können den Bereich leicht einstellen, indem Sie die Parameter auf generate_series() ändern. Es ist möglich, das folgende Konstrukt zu verwenden, um sicherzustellen, dass ranges immer Ihre Ergebnisse decken:

SELECT (ten*10)::text||'-'||(ten*10+9)::text AS range, 
     ten*10 AS r_min, ten*10+9 AS r_max 
    FROM generate_series(0,(SELECT max(score)/10 FROM scores)) AS t(ten)) 

für den ranges CTE.

1

Sie können nicht wie das, aber wenn Sie mit Bereichen abgeleitete Tabelle hinzufügen Dinge möglich:

select ranges.range, count(scores.score) as [number of occurences] 
    from 
    (
    select 0 minRange, 9 maxRange, '0-9' range 
    union all 
    select 10, 19, '10-19' 
    union all 
    select 20, 29, '20-29' 
) ranges 
    left join scores 
    on scores.score between ranges.minRange and ranges.maxRange 
group by ranges.range 

Nicht sicher obwohl über Syntax von postgresql.

EDIT: Haben Sie die Syntax rechts:

select ranges."range", count(scores.score) as "number of occurences" 
    from 
    (
    select 0 minRange, 9 maxRange, '0-9' "range" 
    union all 
    select 10, 19, '10-19' 
    union all 
    select 20, 29, '20-29' 
) as ranges 
    left join scores 
    on scores.score between ranges.minRange and ranges.maxRange 
group by ranges.range 
0
select r.range as [score range], count(*) as [number of occurences] 
from 
    (
    select ' 0- 9' as range, 9 as endrange 
    union all select '10-19',19 
    union all select '20-29',29 
    union all select '30-39',39 
    union all select '40-49',49 
    union all select '50-59',59 
    union all select '60-69',69 
    union all select '70-79',79 
    union all select '80-89',89 
    union all select '90-99',99 
    ) as r 
left join scores s on 
    r.endrange = case 
    when s.score > 90 then 99 
    when s.score > 80 then 89 
    when s.score > 70 then 79 
    when s.score > 60 then 69 
    when s.score > 50 then 59 
    when s.score > 40 then 49 
    when s.score > 30 then 39 
    when s.score > 20 then 29 
    when s.score > 10 then 19 
    when s.score > 0 then 9 
    end 
group by r.range 
0

Eigentlich ist die einfachste Lösung ist, diese (für 2-stellige Nummer):

select substr(rssi::text, 0, 2) || '0-' || substr(rssi::text, 0, 2) || '9' as range, count(*) 
from sensor_stations 
group by substr(rssi::text, 0, 2) 
order by count(*) desc; 

Die Ausgabe in etwa so sein wird:

range | count 
-------+------- 
10-19 | 3414 
30-39 | 1363 
20-29 | 1269 
40-49 | 769 
50-59 | 294 
60-69 | 106 
70-79 |  5 
(7 rows)