是否可以通过ORACLE中的sql查询来解决这种情况?
我有一张这样的桌子:
TYPE UNIT
A 230
B 225
C 60
D 45
E 5
F 2
我需要将单位分成三个(可变)“相同”(大小相等)的间隔,然后 foreach 计算出计数?它的意思是这样的:
0 - 77 -> 4
78 - 154 -> 0
155 - 230 -> 2
是否可以通过ORACLE中的sql查询来解决这种情况?
我有一张这样的桌子:
TYPE UNIT
A 230
B 225
C 60
D 45
E 5
F 2
我需要将单位分成三个(可变)“相同”(大小相等)的间隔,然后 foreach 计算出计数?它的意思是这样的:
0 - 77 -> 4
78 - 154 -> 0
155 - 230 -> 2
您可以使用最大值和连接查询来生成每个范围的上限值和下限值:
select ceil((level - 1) * int) as int_from,
floor(level * int) - 1 as int_to
from (select round(max(unit) / 3) as int from t42)
connect by level <= 3;
INT_FROM INT_TO
---------- ----------
0 76
77 153
154 230
然后对原始表进行左外连接以对每个范围进行计数,这样就可以得到中间范围的零值:
with intervals as (
select ceil((level - 1) * int) as int_from,
floor(level * int) - 1 as int_to
from (select round(max(unit) / 3) as int from t42)
connect by level <= 3
)
select i.int_from || '-' || i.int_to as range,
count(t.unit)
from intervals i
left join t42 t
on t.unit between i.int_from and i.int_to
group by i.int_from, i.int_to
order by i.int_from;
RANGE COUNT(T.UNIT)
---------- -------------
0-76 4
77-153 0
154-230 2
是的,这可以在 Oracle 中完成。困难的部分是边界的定义。您可以在值为 1、2 和 3 的序列上使用最大值和一些算术运算。
之后,剩下的只是一个cross join
和聚合:
with bounds as (
select (case when n = 1 then 0
when n = 2 then trunc(maxu / 3)
else trunc(2 * maxu / 3)
end) as lowerbound,
(case when n = 1 then trunc(maxu / 3)
when n = 2 then trunc(2*maxu / 3)
else maxu
end) as upperbound
from (select 1 as n from dual union all select 2 from dual union all select 3 from dual
) n cross join
(select max(unit) as maxu from atable t)
)
select b.lowerbound || '-' || b.upperbound,
sum(case when units between b.lowerbound and b.upperbound then 1 else 0 end)
from atable t cross join
bounds b
group by b.lowerbound || '-' || b.upperbound;