从数据库

时间:2016-02-20 10:23:52

标签: php mysql date for-loop

我已经通过这里关于这个主题的所有帖子仔细阅读了,但我的问题有一个额外的问题,我需要获得每个房间的床位,因为它的宿舍和一些房间是共享的。

现在我已尝试在PHP中执行此操作,但后来意识到我甚至没有考虑日期范围。所以现在我在想,因为我已经有一个查询,检索在给定日期范围内发生的所有预订,然后将检索到的房间和他们的床与房间表进行比较,并仅显示房间和他们的床没有被占用。但我无法弄清楚如何在床周围工作,因为它们不是一个实体,只是每个房间的床位总数。但是后来预订说房间里的床是预订的。

这是我的表格

房间

enter image description here

预订

enter image description here

现在我用来检索所有预订房间及其床铺的查询是

SELECT rooms_id, bed 
FROM reservations 
WHERE `to` > '2016-02-18' AND `from` < '2016-02-24'

第一个日期,如果我的输入变量$ from和第二个日期是输入变量$ to,它不仅可以检索具有从$ from和$ to开始的预订的房间,还可以检索在日期之前开始的所有预订范围和结束内部,在日期范围之后开始内部和结束,最后在日期范围之前和之后开始的预订。所以上面这个确切的查询将返回下表

enter image description here

然后我可以在我的应用程序中像这样形象化 enter image description here

但这是我被卡住的地方。我不知道如何匹配我的数据,以找到所有可用的房间,但也是床。 从“2016-02-18”到“2016-02-24”的日期范围所需的表格可用房间应如下所示:

|||||||||||||||||||||||||||||||||
|| rooms_id || bed_number ||
||||||||||||||||||||||||||||
||    1     ||    1       ||
----------------------------
||    1     ||    2       ||
----------------------------
||    2     ||    5       ||
----------------------------
||    2     ||    6       ||
----------------------------
||    2     ||    7       ||
----------------------------
||    2     ||    8       ||
----------------------------

你可以在图片中看到这一点,我会在我的应用中展示它的外观。两个日期之间唯一的房间和床位是豪华房及其1号床和2号床以及宿舍床5,6,7,8,因为1-4至少在一个房间预订期望的日期

我唯一的想法是使用NOT IN,但这只有在我不关心床的情况下才有效,而且输出就在这里它是

SELECT *
FROM `rooms`
WHERE `id` NOT IN
     (SELECT rooms_id FROM reservations WHERE `to` > '2016-02-18' AND `from` < '2016-02-24')

enter image description here

而不是我在上面“草绘”

我很感激有关如何处理此问题的任何提示和想法。 我的一部分担心,这一切都归结于我不会把床当作实体而必须这样做,尽管我会永远不会在床上存储任何类型的信息,例如它们的颜色,位置,质量,价格等......

回复@Paul-Spiegel

这太棒了,但是有没有什么方法可以将免费床位作为数字而不是总数。因为当这个人做出预约时,我必须把它分配到床上。所以如果结果可能是

| room_id | title       | beds_total | available_bed_nrs |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
| 1       | luxury room | 2          | 1, 2              |
| 2       | dorm room   | 8          | 5, 6, 7, 8        |

而不是

enter image description here

1 个答案:

答案 0 :(得分:1)

您可以使用此查询获得许多免费房间(以及预订的房间号码):

set @from := '2016-02-18';
set @to   := '2016-02-24';
set @beds := 1;

SELECT rm.id, rm.title, rm.beds,
    rm.beds - IFNULL(rv.num_reserved_beds, 0) AS num_free_beds,
    rv.reserved_bed_nrs
FROM rooms rm
LEFT JOIN (
    SELECT rv.rooms_id, 
        COUNT(1) as num_reserved_beds, 
        GROUP_CONCAT(rv.bed) as reserved_bed_nrs
    FROM reservations rv
    WHERE rv.from < @to
      AND rv.to   > @from
    GROUP BY rv.rooms_id
) rv ON rv.rooms_id = rm.id
HAVING num_free_beds >= @beds

您现在可以解析reserved_bed_nrs,遍历每个房间的所有床位并选择不在reserved_bed_nrs的床位。

<强>阐释:

在日期范围内预订所有床位(不包括):

SELECT *
FROM reservations r
WHERE r.from < @to
  AND r.to   > @from;

按房间分组,计算预留房间的数量,并将所有预留房间数存储在一个字符串字段中:

SELECT rv.rooms_id, 
    COUNT(1) as num_reserved_beds, 
    GROUP_CONCAT(rv.bed) as reserved_bed_nrs
FROM reservations rv
WHERE rv.from < @to
  AND rv.to   > @from
GROUP BY rv.rooms_id

加入(LEFT JOIN)rooms,给定结果计算免费床位数,并将其与您想要预订的床位数进行比较。

更新如何获得免费(非保留)床位:

如果您没有包含所有现有床位的桌子,您将需要某种序列号。假设一个房间最多可以有100张床,您可以创建一个包含100个数字的sequence表:

CREATE TABLE `sequence` (
    `nr` TINYINT(3) UNSIGNED NOT NULL,
    PRIMARY KEY (`nr`)
) select d1.d*10 + d0.d + 1 as nr   from 
    (select 0 d union all select 1 union all select 2 union all select 3 union all select 4 union all select 5 union all select 6 union all select 7 union all select 8 union all select 9) d0,
    (select 0 d union all select 1 union all select 2 union all select 3 union all select 4 union all select 5 union all select 6 union all select 7 union all select 8 union all select 9) d1

现在可以通过交叉加入表roomssequence列出所有现有床位:

SELECT *
FROM rooms rm 
CROSS JOIN sequence seq
WHERE seq.nr <= rm.beds

要列出所有未收到的床位,您可以将其与预订床位的查询相结合(选择所有在预订日期范围内未预订的床位):

SELECT *
FROM rooms rm 
CROSS JOIN sequence seq
WHERE seq.nr <= rm.beds
  AND (rm.id, seq.nr) NOT IN (
        SELECT rv.rooms_id, rv.bed
        FROM reservations rv
        WHERE rv.from < '2016-02-24'
          AND rv.to   > '2016-02-18'
  )

也可以使用NOT EXISTS或排除LEFT JOIN来完成此操作。

您也可以跳过创建sequence表,使用创建代码作为子选择:

SELECT *
FROM rooms rm 
CROSS JOIN (
    select d1.d*10 + d0.d + 1 as nr
    from 
    (select 0 d union all select 1 union all select 2 union all select 3 union all select 4 union all select 5 union all select 6 union all select 7 union all select 8 union all select 9) d0,
    (select 0 d union all select 1 union all select 2 union all select 3 union all select 4 union all select 5 union all select 6 union all select 7 union all select 8 union all select 9) d1
) seq
WHERE seq.nr <= rm.beds
  AND (rm.id, seq.nr) NOT IN (
        SELECT rv.rooms_id, rv.bed
        FROM reservations rv
        WHERE rv.from < '2016-02-24'
          AND rv.to   > '2016-02-18'
  )

http://sqlfiddle.com/#!9/a0d61/5