Sql - группа в диапазонах для включения диапазонов без значений
Предположим, что сценарий похож на этот question. Я хочу получить следующие результаты:
score range | number of occurrences
-------------------------------------
0-9 | 11
10-19 | 14
20-29 | 3
... | ...
И я могу использовать выбранный ответ в качестве решения:
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
Как я могу заверить, что диапазон баллов 30-39 будет отображаться, даже если в этом диапазоне нет результатов?
Ответы
Ответ 1
Попробуйте этот запрос (также на 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:
Вы можете легко отрегулировать диапазон, изменив параметры на generate_series()
. Можно использовать следующую конструкцию, чтобы убедиться, что ranges
всегда будет охватывать ваши оценки:
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))
для ranges
CTE.
Ответ 2
Вам это не понравится, но если вы добавите производную таблицу с диапазонами, станет возможно:
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
Не уверен в синтаксисе postgresql, хотя.
РЕДАКТИРОВАТЬ: Правило синтаксиса:
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
Ответ 3
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
Ответ 4
На самом деле простейшим решением является это (для 2-значных чисел):
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;
Результат будет примерно таким:
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)