我有以下数据:
Product Price StartDate EndDate
Apples 4.9 2010-03-01 00:00:00.000 2010-03-01 00:00:00.000
Apples 4.9 2010-03-02 00:00:00.000 2010-03-02 00:00:00.000
Apples 2.5 2010-03-03 00:00:00.000 2010-03-03 00:00:00.000
Apples 4.9 2010-03-05 00:00:00.000 2010-03-05 00:00:00.000
Apples 4.9 2010-03-06 00:00:00.000 2010-03-06 00:00:00.000
Apples 4.9 2010-03-09 00:00:00.000 2010-03-09 00:00:00.000
Apples 2.5 2010-03-10 00:00:00.000 2010-03-10 00:00:00.000
Apples 4.9 2010-03-11 00:00:00.000 2010-03-11 00:00:00.000
Apples 4.9 2010-03-12 00:00:00.000 2010-03-12 00:00:00.000
Apples 4.9 2010-03-13 00:00:00.000 2010-03-13 00:00:00.000
Apples 4.9 2010-03-15 00:00:00.000 2010-03-15 00:00:00.000
Apples 4.9 2010-03-16 00:00:00.000 2010-03-16 00:00:00.000
想要像product, price, min(startdate), max(startdate)
一样分组,但也应该在开始日期和结束日期进行分组........如下所示
期望的结果
Apples 4.9 2010-03-01 00:00:00.000 2010-03-02 00:00:00.000
Apples 2.5 2010-03-03 00:00:00.000 2010-03-03 00:00:00.000
Apples 4.9 2010-03-05 00:00:00.000 2010-03-09 00:00:00.000
Apples 2.5 2010-03-10 00:00:00.000 2010-03-10 00:00:00.000
Apples 4.9 2010-03-11 00:00:00.000 2010-03-16 00:00:00.000
答案 0 :(得分:3)
SELECT product, price, MIN(start_date), MAX(end_date)
FROM (
SELECT product, price, start_date, end_date,
ROW_NUMBER() OVER (PARTITION BY product ORDER BY startDate) rn1,
ROW_NUMBER() OVER (PARTITION BY product, price ORDER BY startDate) rn2
FROM mytable
) q
GROUP BY
product, price, rn2 - rn1
ORDER BY
product, MIN(start_date), price
答案 1 :(得分:3)
with t2 as
(
select t1.*,
(select count(Price)
from t
where startdate<t1.startdate
and Price<>t1.price
and Product=t1.Product
)
rng
from t as t1
)
select Product,Price,min(startDate),max(EndDate)
from t2 group by Product,Price,RNG
order by 3
答案 2 :(得分:3)
我的方法。
数据:
create table t ( producte varchar(50),
price money,
start_date date,
end_date date);
insert into t values
( 'apple', 4.9, '2012-01-01', '2012-01-01' ),
( 'apple', 4.9, '2012-01-02', '2012-01-02' ),
( 'apple', 8, '2012-01-04', '2012-01-04' ),
( 'cat', 5, '2012-01-01', '2012-01-01' ),
( 'cat', 6, '2012-01-02', '2012-01-02' ),
( 'cat', 6, '2012-01-03', '2012-01-03' );
查询:
with start_dates as (
select
t.producte, t.price, t.start_date, t.end_date, t.start_date as gr_date
from
t left outer join
t t1 on
t.price = t1.price and --new
t.producte = t1.producte and
t.start_date = dateadd(day,1, t1.end_date )
where t1.producte is null
union all
select
t.producte, t.price, t.start_date,t. end_date, gr_date
from
t inner join
start_dates t1 on
t.price = t1.price and --new
t.producte = t1.producte and
t.start_date = dateadd(day,1, t1.end_date )
)
select t.producte, t.price , min( t.start_date ), max( t.end_date )
from start_dates t
group by t.producte, gr_date ,t.price
| PRODUCTE | PRICE | COLUMN_2 | COLUMN_3 |
----------------------------------------------
| apple | 4.9 | 2012-01-01 | 2012-01-02 |
| apple | 8 | 2012-01-04 | 2012-01-04 |
| cat | 5 | 2012-01-01 | 2012-01-01 |
| cat | 6 | 2012-01-02 | 2012-01-03 |
<强>解释强>
这是一个递归的CTE表达式。基本查询采用每组价格的初始日期。递归查询以此价格查找最后一个数据。
答案 3 :(得分:1)
以下是一个建议:对于每一行,您必须找到价格不同的最大上一个日期,并对此进行分组。例如,对于2010-03-11和2010-03-16之间的任何行,您必须检索日期2010-03-10,因为这是价格不同的最大上一个日期(2.5对4.9)。第一行将返回一个空日期,但这应该不是问题。
但是,对于很长的表,这种查询可能会变得非常慢。因此,如果你有一些速度问题,你应该考虑添加一个列并使用光标逐步填充它的可能性:你按日期循环它,每次看到一个新的价格,你改变它的值。最后的分组是微不足道的。
这是:
Select Product, Price, Min(StartDate) as StartDate, PreviousDate from (
Select product, price, StartDate, (Select max (StartDate) from table_2 t3 where t3.price <> t2.price and t3.StartDate < t2.StartDate and t3.Product = t2.Product) as previousDate
from table_2 t2) SQ
Group by Product, Price, PreviousDate
Order by PreviousDate
答案 4 :(得分:0)
我相信这是迄今为止表现最佳的解决方案:
WITH Calc AS (
SELECT *,
Grp = DateAdd(day, -Row_Number()
OVER (PARTITION BY Product, Price ORDER BY StartDate), StartDate
)
FROM dbo.PriceHistory
)
SELECT Product, Price, FromDate = Min(StartDate), ToDate = Max(StartDate)
FROM Calc
GROUP BY Product, Price, Grp
ORDER BY FromDate;