Oracle
 sql >> Teknologi Basis Data >  >> RDS >> Oracle

Dapatkan jam kelas yang tersedia antara rentang waktu tanggal

Saya sedang mencari solusi yang serupa dalam konsep setidaknya dengan Wernfried, tetapi saya pikir itu cukup berbeda untuk diposting juga. Awalannya adalah ide yang sama, pertama-tama menghasilkan kemungkinan slot waktu, dan dengan asumsi Anda melihat jendela 15 menit:Saya menggunakan CTE karena menurut saya mereka lebih jelas daripada pilihan bersarang, terutama dengan banyak level ini.

with date_time_range as (
  select to_date('10/10/2013 07:00', 'DD/MM/YYYY HH24:MI') as date_start,
    to_date('10/10/2013 21:15', 'DD/MM/YYYY HH24:MI') as date_end
  from dual
),
time_slots as (
  select level as slot_num,
    dtr.date_start + (level - 1) * interval '15' minute as slot_start,
    dtr.date_start + level * interval '15' minute as slot_end
  from date_time_range dtr
  connect by level <= (dtr.date_end - dtr.date_start) * (24 * 4) -- 15-minutes
)
select * from time_slots;

Ini memberi Anda 57 slot 15 menit antara tanggal mulai dan akhir yang Anda tentukan. CTE untuk date_time_range tidak sepenuhnya diperlukan, Anda dapat memasukkan tanggal langsung ke time_slots kondisi, tetapi Anda harus mengulanginya dan kemudian memperkenalkan kemungkinan titik kegagalan (dan berarti mengikat nilai yang sama beberapa kali, dari JDBC atau di mana pun).

Slot tersebut kemudian dapat digabungkan ke daftar ruang kelas, yang saya asumsikan sudah ada di tabel lain, yang memberi Anda 171 (3x57) kombinasi; dan itu dapat dibandingkan dengan pemesanan yang ada - setelah itu dihilangkan, Anda memiliki 153 slot 15 menit yang tidak memiliki pemesanan.

with date_time_range as (...),
time_slots as (...),
free_slots as (
  select c.classroom, ts.slot_num, ts.slot_start, ts.slot_end,
    lag(ts.slot_end) over (partition by c.classroom order by ts.slot_num)
      as lag_end,
    lead(ts.slot_start) over (partition by c.classroom order by ts.slot_num)
      as lead_start
  from time_slots ts
  cross join classrooms c
  left join occupied_classrooms oc on oc.classroom = c.classroom
    and not (oc.occupied_end <= ts.slot_start 
      or oc.occupied_start >= ts.slot_end)
  where oc.classroom is null
)
select * from free_slots;

Tapi kemudian Anda harus menciutkannya ke dalam rentang yang berdekatan. Ada berbagai cara untuk melakukan itu; di sini saya mengintip baris sebelumnya dan berikutnya untuk memutuskan apakah nilai tertentu adalah tepi rentang:

with date_time_range as (...),
time_slots as (...),
free_slots as (...),
free_slots_extended as (
  select fs.classroom, fs.slot_num,
    case when fs.lag_end is null or fs.lag_end != fs.slot_start
      then fs.slot_start end as slot_start,
    case when fs.lead_start is null or fs.lead_start != fs.slot_end
      then fs.slot_end end as slot_end
  from free_slots fs
)
select * from free_slots_extended
where (fse.slot_start is not null or fse.slot_end is not null);

Sekarang kita turun menjadi 12 baris. (Bagian luar where klausa menghilangkan semua 141 dari 153 slot dari langkah sebelumnya yang merupakan mid-range, karena kami hanya peduli dengan tepinya):

CLASSROOM   SLOT_NUM SLOT_START       SLOT_END       
--------- ---------- ---------------- ----------------
A                  1 2013-10-10 07:00                  
A                 12                  2013-10-10 10:00 
A                 19 2013-10-10 11:30                  
A                 57                  2013-10-10 21:15 
B                  1 2013-10-10 07:00                  
B                  9                  2013-10-10 09:15 
B                 16 2013-10-10 10:45                  
B                 30                  2013-10-10 14:30 
B                 37 2013-10-10 16:00                  
B                 57                  2013-10-10 21:15 
C                  1 2013-10-10 07:00                  
C                 57                  2013-10-10 21:15 

Jadi itu mewakili tepi, tetapi pada baris terpisah, dan langkah terakhir menggabungkannya:

...
select distinct fse.classroom,
  nvl(fse.slot_start, lag(fse.slot_start)
    over (partition by fse.classroom order by fse.slot_num)) as slot_start,
  nvl(fse.slot_end, lead(fse.slot_end)
    over (partition by fse.classroom order by fse.slot_num)) as slot_end
from free_slots_extended fse
where (fse.slot_start is not null or fse.slot_end is not null)

Atau menggabungkan semua itu:

with date_time_range as (
  select to_date('10/10/2013 07:00', 'DD/MM/YYYY HH24:MI') as date_start,
    to_date('10/10/2013 21:15', 'DD/MM/YYYY HH24:MI') as date_end
  from dual
),
time_slots as (
  select level as slot_num,
    dtr.date_start + (level - 1) * interval '15' minute as slot_start,
    dtr.date_start + level * interval '15' minute as slot_end
  from date_time_range dtr
  connect by level <= (dtr.date_end - dtr.date_start) * (24 * 4) -- 15-minutes
),
free_slots as (
  select c.classroom, ts.slot_num, ts.slot_start, ts.slot_end,
    lag(ts.slot_end) over (partition by c.classroom order by ts.slot_num)
      as lag_end,
    lead(ts.slot_start) over (partition by c.classroom order by ts.slot_num)
      as lead_start
  from time_slots ts
  cross join classrooms c
  left join occupied_classrooms oc on oc.classroom = c.classroom
    and not (oc.occupied_end <= ts.slot_start
      or oc.occupied_start >= ts.slot_end)
  where oc.classroom is null
),
free_slots_extended as (
  select fs.classroom, fs.slot_num,
    case when fs.lag_end is null or fs.lag_end != fs.slot_start
      then fs.slot_start end as slot_start,
    case when fs.lead_start is null or fs.lead_start != fs.slot_end
      then fs.slot_end end as slot_end
  from free_slots fs
)
select distinct fse.classroom,
  nvl(fse.slot_start, lag(fse.slot_start)
    over (partition by fse.classroom order by fse.slot_num)) as slot_start,
  nvl(fse.slot_end, lead(fse.slot_end)
    over (partition by fse.classroom order by fse.slot_num)) as slot_end
from free_slots_extended fse
where (fse.slot_start is not null or fse.slot_end is not null)
order by 1, 2;

Yang memberikan:

CLASSROOM SLOT_START       SLOT_END       
--------- ---------------- ----------------
A         2013-10-10 07:00 2013-10-10 10:00 
A         2013-10-10 11:30 2013-10-10 21:15 
B         2013-10-10 07:00 2013-10-10 09:15 
B         2013-10-10 10:45 2013-10-10 14:30 
B         2013-10-10 16:00 2013-10-10 21:15 
C         2013-10-10 07:00 2013-10-10 21:15 

SQL Fiddle .



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Memetakan satu kelas Entitas ke dua database yang berbeda (Oracle dan Ingres)

  2. Sql Pesan dengan di beberapa kolom

  3. perulangan melalui array untuk kondisi where pl/sql

  4. Oracle SQL -- hapus sebagian duplikat dari string

  5. Menggunakan Oracle dan PHP:Bekerja di SQL Developer tetapi hasil file PHP ORA-00900:Pernyataan Tidak Valid