根据标准费率和季节性日期范围计算价格

时间:2015-03-11 14:20:22

标签: php mysql sql datetime date-range

我正在建立预订风格系统。房间有年度标准费率(这是我的问题与其他类似的不同,因为其他人有独特的范围)然后他们也有一年中的日期范围,设置不同的价格。我想要计算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,我想要的是如下计算:

  • 2015-07-15 ... 2015-07-19以标准费率(每晚100人)
  • 2015-07-20 ... 2015-07-24夏季率(每晚150人)

哪个应该总计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

3 个答案:

答案 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)

不是最优雅的解决方案,但它可以工作,而且速度快,可以让你使用连接而不是循环查询,我更喜欢。