我正在建立预订风格系统。房间有年度标准费率(这是我的问题与其他类似的不同,因为其他人有独特的范围)然后他们也有一年中的日期范围,设置不同的价格。我想要计算MYSQL中的价格,但是当用户选择特殊日期范围内的日期时,它会将该价格相加,但也会将这些日期的标准价格相加,这意味着总数太高。这有点罗嗦,所以这里是查询和一个例子:
SELECT SUM(Price * (1 + DATEDIFF(LEAST(End_date, '2015-07-25' - INTERVAL 1 DAY), GREATEST(Start_date, '2015-07-15')))) AS Total
FROM room_rates
WHERE roomId = '46' AND (
'2015-07-25' - INTERVAL 1 DAY BETWEEN Start_date AND End_date
OR '2015-07-15' BETWEEN Start_date AND End_date
)
+--------+------------+------------+------------+------+
| RoomId | Range Name | Start_Date | End_Date | Rate |
+--------+------------+------------+------------+------+
| 46 | Standard | 2015-01-01 | 2015-12-31 | 100 |
+--------+------------+------------+------------+------+
| 46 | Summer | 2015-07-20 | 2015-08-31 | 150 |
+--------+------------+------------+------------+------+
| 46 | Christmas | 2015-12-18 | 2015-12-31 | 180 |
+--------+------------+------------+------------+------+
如果用户在我的查询中选择2015-07-15到2015-07-25,我想要的是如下计算:
哪个应该总计1250. 但是因为标准费率的日期介于一年的第一天和最后一天之间,所以查询还包括夏季日期间该范围内的价格和汇总费率,这意味着我按标准费率收取所有10个日期,加上夏季费用的5晚,总计1750。
所以我的问题是,如果没有替代方案,我怎样才能将查询修改为仅使用标准费率?标准费率总是被称为“标准”,所以我可以轻松识别它们,我只是不知道要做出什么改变!
修改
我应该补充一下,我希望在PHP内部这样做(我正在使用PDO)
第二次修改 另外值得注意的是,日期范围总是在一年内,并且没有日期范围可以重叠(除了占用整年的标准费率)
解
决定改变我的方法,并按照几个人建议的答案,建立一个日常的数据库。这是我最后的工作查询。感谢大家的帮助和建议。
SELECT standard_rates.villaId as `villaId`,
sum(IFNULL(custom_rates.nightly_rate_usd, standard_rates.nightly_rate_usd))
AS `Rate`
FROM dates
LEFT JOIN
villa_price_bands AS standard_rates
ON standard_rates.Name = 'Standard'
AND dates.date BETWEEN standard_rates.Start_Date AND standard_rates.End_Date
AND FIND_IN_SET(standard_rates.villaId, :resultIds)
LEFT JOIN
villa_price_bands AS custom_rates
ON custom_rates.Name != 'Standard'
AND dates.date BETWEEN custom_rates.Start_Date AND custom_rates.End_Date
AND custom_rates.villaId = standard_rates.villaId
WHERE dates.date >= :arrDate
AND dates.date < :deptDate
GROUP BY villaId
答案 0 :(得分:1)
我会使用一个日期表:一个表(datelist
),由一列(date
)组成,包含前一个和即将到来的 n 年的所有日期
查询的粗略轮廓(您可能需要更正结束日期):
SELECT
datelist.date AS Night,
IFNULL(seasonal_rates.Range_Name, standard_rates.Range_Name) AS Season,
IFNULL(seasonal_rates.Rate, standard_rates.Rate) AS Rate
FROM datelist
LEFT JOIN rates AS standard_rates ON standard_rates.Range_Name = 'Standard'
LEFT JOIN rates AS seasonal_rates ON datelist.date BETWEEN seasonal_rates.Start_Date AND seasonal_rates.End_Date
WHERE standard_rates.RoomId = 46
AND seasonal_rates.RoomId = 46
AND datelist.date BETWEEN '2015-07-15' AND '2015-07-25'
然后,您可以将结果传递给SUM / GROUP BY查询。
答案 1 :(得分:0)
使用MySQL CASE语句。
例如
SELECT Columnname,
sum(CASE
WHEN RangeName LIKE '%Standard%'
THEN do something END) as AliasName
FROM YourTable
答案 2 :(得分:0)
我之前处理这类问题的方法是建立一个'日期'表。
1/1/1990 .. 1/1/2050(你可以在飞行中做到这一点,但是为了几Mb的存储空间,从'真实'表中获得的性能提升值得设置一次。
例如。
|Date ID| Date |
|---------------------|
|0000001| 2015-01-01 |
|0000002| 2015-01-02 |
然后您可以基于此构建视图。如:
Select
[date id],
if(range_name='standard',rate,0) standardPrices,
if(range_name='Summer',rate,0) summerPrices,
if(range_name='Winter',rate,0) winterPrices
from
date_table left join room_rates
on date_table.date between start_date and end_date
where room = '46' and date_table.date between '2015-07-15' and '2015-07-25'
给你一个像
这样的观点date |standardPrices|summerPrices|winterPrices|
------------------------------------------------------
2015-07-15 | 100 | 0 | 0 |
2015-07-16 | 100 | 0 | 0 |
2015-07-17 | 100 | 0 | 0 |
2015-07-18 | 100 | 0 | 0 |
2015-07-19 | 100 | 0 | 0 |
2015-07-20 | 100 | 0 | 0 |
2015-07-20 | 0 | 150 | 0 |
2015-07-21 | 100 | 0 | 0 |
2015-07-21 | 0 | 150 | 0 |
如果你那么把这个观点分组为'日期;您可以反过来使用它来运行查询聚合(再次使用'if(summerPrices&gt; 0,summerPrices,standardPrices)
不是最优雅的解决方案,但它可以工作,而且速度快,可以让你使用连接而不是循环查询,我更喜欢。