Задать вопрос
@del993788

Как проверить повторения в бд и выдать номер?

Здравствуйте. Пишу скрипт для бронирования номеров в гостинице.
Есть две вот такие таблицы:
allnomer sqlfiddle.com/#!9/75102 , где указаны все номера. В гостинице есть 3 номера lux: 301, 302 и 303
и main sqlfiddle.com/#!9/02c4e , где указаны все брони.
Я пытаюсь сделать скрипт, который будет проверять номера на занятость, и в случае нахождения свободного номера, выдаёт брони этот номер.
Чуть подробнее. У меня есть дата заезда и дата выезда новой брони: 2016-11-28 и 2016-11-30. Это диапазон дат. То есть 28, 29 и 30 числа в гостинице собираются жить люди. В гостинице уже забронированы два номера. Первый (301) на даты с 2016-11-29 по 2016-12-01, второй (302) на даты с 2016-12-01 по 2016-12-02. Человек не может заехать в 301 номер, так как 29 и 30 числа этот номер будет занят. У нас остаётся два номера: 302 и 303. 302 номер занят только 01 и 02. Ни одна дата не пересекается и поэтому нам 302 номер подходит. Переменная $namenomer принимает значение 302 и подставляется в скрипт.
Подскажите, как сделать проверку на свободность номера, чтобы, в случае хотя бы одного совпадения, проверялся следующий номер? Вероятность того, что на этом этапе не найдётся ни одного свободного номера исключена. До этого этапа уже есть скрипт, который проверяет все номера на свободность. Однако он проверяет все номера. Этот же скрипт нужен для проверки определённых номеров.

Вот такой у меня сейчас код:
SELECT a.nomer
FROM allnomer a
LEFT JOIN main m ON
a.nomer = m.numbernomer AND (
m.datestart BETWEEN '2016-11-28' AND '2016-11-30' OR
m.dateend BETWEEN '2016-11-28' AND '2016-11-30' )
WHERE a.TYPE = 'lux' AND m.id IS NULL


И эти даты он обрабатывает правильно. Я получаю ответ 302.

Но если структура таблицы main вот такая sqlfiddle.com/#!9/fba1bd
А запрос вот такой:
SELECT a.nomer
FROM allnomer a
LEFT JOIN main m ON
a.nomer = m.numbernomer AND (
m.datestart BETWEEN '2016-12-23' AND '2016-12-23' OR
m.dateend BETWEEN '2016-12-23' AND '2016-12-23' )
WHERE a.TYPE = 'lux' AND m.id IS NULL


То возвращается 301, хотя 23 число находится в диапазоне чисел таблицы main.
Тоже самое со всеми числами, которые находятся в этом промежутке, кроме 19 и 25 числа.
  • Вопрос задан
  • 237 просмотров
Подписаться 2 Оценить 1 комментарий
Пригласить эксперта
Ответы на вопрос 1
sergey-gornostaev
@sergey-gornostaev
Седой и строгий
Во-первых, стоит повысить нормализацию:
create table rooms (id integer primary key, type varchar(7), number integer);

insert into rooms
    (id, type, number)
values
    (1, 'lux', 301),
    (2, 'lux', 302),
    (3, 'lux', 303);

create table booking (
  id integer primary key,
  start_date date,
  end_date date,
  room integer,
  foreign key(room) references rooms(id)
);
	
insert into booking (id, start_date, end_date, room)
values
    (1, '2016-11-29', '2016-12-01', 1),
    (2, '2016-12-01', '2016-12-02', 2),
    (3, '2016-12-02', '2016-12-02', 1),
    (4, '2016-12-02', '2016-12-02', 3);

Во-вторых, проверить перекрытие диапазонов дат можно проще:
select number from rooms
where id not in (
    select room from booking
    where start_date > '2016-11-28' and start_date < '2016-11-30'
) and type = 'lux'
order by number
limit 1;

Вернёт первый по порядку номер не забронированный ...
Ответ написан
Ваш ответ на вопрос

Войдите, чтобы написать ответ

Похожие вопросы