Текущее время между двумя заданными временами?
Я пытаюсь вычислить , если текущее время находится в пределах часа работы ресторана.
Этот вопрос был задан очень много на Stackoverflow, но я не нашел тот, который может объяснить проблемы, которые у меня возникают. Кроме того, было бы неплохо увидеть идею о том, как это сделать.
В настоящее время он ломается, если день закрыт (воскресенье в этом примере), или если он включен в "субботу" (так технически 1 утра в воскресенье). У меня есть чувство, что мне придется изменить способ хранения данных для учетной записи после полуночи, но я пытаюсь работать с тем, что у меня есть сейчас. Это проблема, потому что большинство ресторанов перечисляют свое время открытия в течение заданного дня с 5:00 до 2:00, а не с 17:00 до 12:00, 12:00 - 2:00.
В любом случае, вот что я имею. Скажите, пожалуйста, лучший способ сделать это.
У меня есть время, подобное этому:
$times = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '9am - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '5:30pm - 2am',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
Это код, который я использую сейчас:
// Get the right key for today
$status = 'open';
$now = (int) current_time( 'timestamp' );
$day = strtolower( date('D', $now) );
$string = 'opening_hours_'.$day;
$times = $meta[$string][0]; // This should be a stirng like '6:00am - 2:00am' or even '6:00am - 11:00am, 1:00pm to 11:00pm'.
// Does it contain a '-', if not assume it closed.
$pos = strpos($times, '-');
if ($pos === false) {
$status = 'closed';
} else {
// Maybe a day has multiple opening times?
$seating_times = explode(',', $times);
foreach( $seating_times as $time ) {
$chunks = explode('-', $time);
$open_time = strtotime($chunks[0]);
$close_time = strtotime($chunks[1]);
// Calculate if now is between range of open and closed
if(($open_time <= $now) && ($now <= $close_time)) {
$status = 'open';
break;
} else {
$status = 'closed';
}
}
}
ПРИМЕЧАНИЕ: current_time ('timestamp', 0) является функцией WordPress.
Ответы
Ответ 1
Вот мое объектно-ориентированное решение, основанное на использовании PHP класса DateTime (доступно со версии 5.2):
<?php
class Restaurant {
private $cw;
private $times = array();
private $openings = array();
public function __construct(array $times) {
$this->times = $times;
$this->setTimes(date("w") ? "this" : "last");
//print_r($this->openings); // Debug
}
public function setTimes($cw) {
$this->cw = $cw;
foreach ($this->times as $key => $val) {
$t = array();
$buf = strtok($val, ' -,');
for ($n = 0; $buf !== FALSE; $n++) {
try {
$d = new DateTime($buf);
$d->setTimestamp(strtotime(substr($key, -3)." {$this->cw} week {$buf}"));
if ($n && ($d < $t[$n-1])) {
$d->add(new DateInterval('P1D'));
}
$t[] = $d;
} catch (Exception $e) {
break;
}
$buf = strtok(' -,');
}
if ($n % 2) {
throw new Exception("Invalid opening time: {$val}");
} else {
$this->openings[substr($key, -3)] = $t;
}
}
}
public function isOpen() {
$cw = date("w") ? "this" : "last";
if ($cw != $this->cw) {
$this->setTimes($cw);
}
$d = new DateTime('now');
foreach ($this->openings as $wd => $t) {
$n = count($t);
for ($i = 0; $i < $n; $i += 2) {
if (($d >= $t[$i]) && ($d <= $t[$i+1])) {
return(TRUE);
}
}
}
return(FALSE);
}
}
$times = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '9am - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '9am - 3pm',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
try {
$r = new Restaurant($times);
$status = $r->isOpen() ? 'open' : 'closed';
echo "status=".$status.PHP_EOL;
} catch (Exception $e) {
echo $e->getMessage().PHP_EOL;
}
?>
Как вы можете видеть, конструктор строит внутреннюю форму (массив openings
объектов DateTime), который затем используется с простым сравнением в методе isOpen
, чтобы проверить, есть ли во время вызова ресторан открывается или закрывается.
Вы также заметите, что я использовал метод DateTime: add для вычисления даты завтрашнего дня вместо добавления 86400 (24 * 60 * 60) до отметки времени текущей даты, чтобы избежать проблем с DST смены времени.
Доказательство концепции:
<?php
ini_set("date.timezone", "Europe/Rome");
echo "date.timezone = ".ini_get("date.timezone").PHP_EOL;
$d1 = strtotime("2013-10-27 00:00:00");
$d2 = strtotime("2013-10-28 00:00:00");
// Expected: 86400, Result: 90000
echo "Test #1: ".($d2 - $d1).PHP_EOL;
// Expected: 2013-10-28 00:00:00, Result: 2013-10-27 23:00:00
echo "Test #2: ".date("Y-m-d H:i:s", $d1 + 86400).PHP_EOL;
$d1 = strtotime("2014-03-30 00:00:00");
$d2 = strtotime("2014-03-31 00:00:00");
// Expected: 86400, Result: 82800
echo "Test #3: ".($d2 - $d1).PHP_EOL;
// Expected: 2014-03-30 00:00:00, Result: 2014-03-29 23:00:00
echo "Test #4: ".date("Y-m-d H:i:s", $d2 - 86400).PHP_EOL;
?>
Что дает следующие результаты:
date.timezone = Europe/Rome
Test #1: 90000
Test #2: 2013-10-27 23:00:00
Test #3: 82800
Test #4: 2014-03-29 23:00:00
Поэтому кажется, что в один прекрасный день не всегда есть 86400 секунд; по крайней мере, не два раза в год...
Ответ 2
Предположим, что вместо такого массива у нас был другой элемент со следующими типами записей:
Array ( [from] => 1382335200 [to] => 1382374800 )
Значения from
и to
- это отметки времени, рассчитанные путем проецирования информации о вашем массиве на текущую (рабочую) неделю.
Затем, чтобы проверить, открыт ли ресторан прямо сейчас, нам нужно было сделать что-то просто:
$slots=..... /* calculate time slots array */
$status='closed';
$rightnow=time();
foreach($slots as $slot)
if($rightnow<=$slot['to'])
{
if($rightnow>=$slot['from']) $status='open';
break;
}
echo "The restaurant is <strong>$status</strong> right now<br>";
Учитывая день недели, в виде mon
, tue
, wed
и т.д. и двух строк, которые определяют временной диапазон, например 8:30am
и 3:15pm
, следующая функция вернет соответствующее время слот, как описано выше:
function get_time_slot($weekday,$fromtime,$totime)
{
$from_ts=strtotime("this week $weekday $fromtime");
$to_ts=strtotime("this week $weekday $totime");
if($to_ts<$from_ts)
{
$to_ts=strtotime("this week $weekday +1 day $totime");
if($to_ts>strtotime("next week midnight"))
$to_ts=strtotime("this week mon $totime");
}
return array('from'=>$from_ts,'to'=>$to_ts);
}
strtotime()
может творить чудеса, а? Обратите внимание, что если конец временного интервала окажется раньше начала, мы предполагаем, что он относится к следующему дню, и мы переучитываем его как таковой.
РЕДАКТИРОВАТЬ: Сначала я наивно думал, что исправлю его, добавив день в считанные секунды. Это было не совсем точно, так как манипулирование меткой времени не сохраняет информацию о ДСТ. Поэтому, если в слот включен дневной сдвиг (полуночи), а также сдвиг DST, он даст неточные результаты на час. Используя strtotime()
снова и снова, с тем же аргументом плюс день, установите его прямо.
yaEDIT: Исправлена еще одна ошибка (надеюсь, последняя): когда ресторан открыт по воскресеньям до полуночи, $to_time
должен записаться на эту неделю в понедельник, в то же время. Уф!
Теперь, чтобы преобразовать массив, вам нужно будет сделать:
$slots=array();
foreach($times as $key=>$entry)
{
list(,,$dow)=explode('_',$key);
foreach(explode(',',$entry) as $d)
{
$arr=explode('-',$d);
if(count($arr)==2) $slots[]=get_time_slot($dow,$arr[0],$arr[1]);
}
}
Здесь немного phpfiddle, чтобы продемонстрировать это.
РЕДАКТИРОВАТЬ. Мотивированный обсуждением "краткости" в другом ответе, я думал, что дам свою "компактную" версию. Используя ту же логику, она сводится к следующему:
$status='closed';
$rightnow=time();
foreach($times as $key=>$entry)
{
list(,,$dow)=explode('_',$key);
foreach(explode(',',$entry) as $d)
if(count($arr=explode('-',$d))==2)
{
$from_ts=strtotime("this week $dow {$arr[0]}");
$to_ts=strtotime("this week $dow {$arr[1]}");
if($to_ts<$from_ts) $to_ts=strtotime("this week $dow +1 day {$arr[1]}");
{
$to_ts=strtotime("this week $dow +1 day {$arr[1]}");
if($to_ts>strtotime("next week midnight"))
$to_ts=strtotime("this week mon {$arr[1]}");
}
if($rightnow<=$to_ts)
{
if($rightnow>=$from_ts) $status='open';
break 2; // break both loops
}
}
}
echo "<hr>The restaurant is <strong>$status</strong> right now<br>";
Тем не менее, я сам по-прежнему предпочитаю оригинальную версию. Помимо очевидных преимуществ наличия функции, массив $slots
может быть очень хорошо кэширован и повторно использован, что делает соответствующие вычисления намного проще, чем разбор исходных данных снова и снова.
Ответ 3
Это, вероятно, не самый эффективный, но он должен хорошо работать для вашей проблемы:
$times = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '9am - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '9am - 3pm',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
var_dump(is_open($times, strtotime('sun 1am'))); // true
Здесь первая функция, простая в дизайне; он использует сетку времени открытия и закрытия и определяет, соответствует ли данное время любому диапазону:
function is_open($times, $now)
{
$today = strtotime('today', $now);
$grid = get_time_grid($times);
$today_name = strtolower(date('D', $today));
$today_seconds = $now - $today;
foreach ($grid[$today_name] as $range) {
if ($today_seconds >= $range[0] && $today_seconds < $range[1]) {
return true;
}
}
return false;
}
Эта функция создает реальную сетку; если заканчивается диапазон до его соответствующего запуска, он создаст два диапазона: по одному на каждый день, который будет растянут.
function get_time_grid($times)
{
static $next_day = array(
'mon' => 'tue', 'tue' => 'wed', 'wed' => 'thu',
'thu' => 'fri', 'fri' => 'sat', 'sat' => 'sun',
'sun' => 'mon'
);
static $time_r = '(\d{1,2}(?::\d{2})?(?:am|pm))';
$today = strtotime('today');
$grid = [];
foreach ($times as $key => $schedule) {
$day_name = substr($key, -3);
// match all time ranges, skips "closed"
preg_match_all("/$time_r - $time_r/", $schedule, $slots, PREG_SET_ORDER);
foreach ($slots as $slot) {
$from = strtotime($slot[1], $today) - $today;
$to = strtotime($slot[2], $today) - $today;
if ($to < $from) { // spans two days
$grid[$day_name][] = [$from, 86400];
$grid[$next_day[$day_name]][] = [0, $to];
} else { // normal range
$grid[$day_name][] = [$from, $to];
}
}
}
return $grid;
}
В коде есть только несколько комментариев, но я надеюсь, что вы сможете следить за тем, что делается. Дайте мне знать, если вам нужно какое-либо разъяснение.
Ответ 4
Неделя
$times = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '9am - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '9am - 3pm',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
Пример использования:
ini_set( "date.timezone", "Pacific/Auckland" ); // Make sure correct timezone is set
echo ( isOpen( $times ) ? 'Open' : 'Closed' );
echo ( isOpen( $times ,"10am" ) ? 'Open' : 'Closed' );
Определение функции:
/*
* First parameter : Weeks timings as array
* Second parameter : Time to check as string
* Return value : boolean
*/
function isOpen( $times ,$timeToCheck = 'now' )
{
$timeToCheckAsUnixTimestamp = strtotime( $timeToCheck );
$yesterdayTimes = $todayTimes = '';
// Find yesterday times and today times
foreach( $times as $day => $timeRange )
{
$yesterdayTimes = ( stripos( $day ,date( "D" ,time() - 60 * 60 * 24 ) ) !== false ? $timeRange : $yesterdayTimes );
$todayTimes = ( stripos( $day ,date( "D" ) ) !== false ? $timeRange : $todayTimes );
}
// Handle closed
if( strcasecmp( $todayTimes ,'closed' ) == 0 ) return false;
if( strcasecmp( $yesterdayTimes ,'closed' ) == 0 ) $yesterdayTimes = '12am - 12am';
// Process and check with yesterday timings
foreach( explode( ',' ,$yesterdayTimes ) as $timeRanges )
{
list( $from ,$to ) = explode( '-' ,$timeRanges );
list( $fromAsUnixTimestamp ,$toAsUnixTimestamp ) = array( strtotime( $from .' previous day' ) ,strtotime( $to .' previous day' ) );
$toAsUnixTimestamp = ( $toAsUnixTimestamp < $fromAsUnixTimestamp ? strtotime( $to ) : $toAsUnixTimestamp );
if( $fromAsUnixTimestamp <= $timeToCheckAsUnixTimestamp and $timeToCheckAsUnixTimestamp <= $toAsUnixTimestamp ) return true;
}
// Process and check with today timings
foreach( explode( ',' ,$todayTimes ) as $timeRanges )
{
list( $from ,$to ) = explode( '-' ,$timeRanges );
list( $fromAsUnixTimestamp ,$toAsUnixTimestamp ) = array( strtotime( $from ) ,strtotime( $to ) );
$toAsUnixTimestamp = ( $toAsUnixTimestamp < $fromAsUnixTimestamp ? strtotime( $to .' next day' ) : $toAsUnixTimestamp );
if( $fromAsUnixTimestamp <= $timeToCheckAsUnixTimestamp and $timeToCheckAsUnixTimestamp <= $toAsUnixTimestamp ) return true;
}
return false;
}
Ответ 5
Я сделал что-то подобное в прошлом, но использовал совершенно другой подход. Я хранил часы работы в отдельной таблице.
CREATE TABLE `Openinghours`
(
`OpeninghoursID` int, // serial
`RestaurantID` int, // foreign key
`Dayofweek` int, // day of week : 0 (for Sunday) through 6 (for Saturday)
`Opentime` int, // time of day when restaurant opens (in seconds)
`Closetime` int // time of day when restaurant closes (in seconds)
);
Если в ресторане имеется более одного периода открытия в день, вы можете просто добавить 2 записи (или больше идентификаторов). Приятно использовать такую таблицу, чтобы вы могли просто запросить, какие рестораны открыты.
$day = date('w');
$now = time()-strtotime("00:00");
$query = "Select `RestaurantID` from `Openinghours` where `Dayofweek` = ".$day." and `Opentime` <= ".$now." and `Closetime` > ".$now;
Еще одна красота использования такой системы заключается в том, что вы можете адаптировать ваш запрос для получения разных результатов, например, например, какие рестораны открыты сейчас и оставаться открытыми в течение как минимум еще одного часа (нет смысла ходить в ресторан за несколько минут до закрытия )
$day = date('w');
$now = time()-strtotime("00:00");
$query = "Select `RestaurantID` from `Openinghours` where `Dayofweek` = ".$day." and `Opentime` <= ".$now." and `Closetime` > ".($now+3600);
Конечно, он нуждается в переформатировании ваших текущих данных, но он приносит приятные функции.
Ответ 6
Вот еще одно решение, не требующее переформатировать ваши данные.
$times = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '9am - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '5:30pm - 2am',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
function compileHours($times, $timestamp) {
$times = $times['opening_hours_'.strtolower(date('D',$timestamp))];
if(!strpos($times, '-')) return array();
$hours = explode(",", $times);
$hours = array_map('explode', array_pad(array(),count($hours),'-'), $hours);
$hours = array_map('array_map', array_pad(array(),count($hours),'strtotime'), $hours, array_pad(array(),count($hours),array_pad(array(),2,$timestamp)));
end($hours);
if ($hours[key($hours)][0] > $hours[key($hours)][1]) $hours[key($hours)][1] = strtotime('+1 day', $hours[key($hours)][1]);
return $hours;
}
function isOpen($now, $times) {
$open = 0; // time until closing in seconds or 0 if closed
// merge opening hours of today and the day before
$hours = array_merge(compileHours($times, strtotime('yesterday',$now)),compileHours($times, $now));
foreach ($hours as $h) {
if ($now >= $h[0] and $now < $h[1]) {
$open = $h[1] - $now;
return $open;
}
}
return $open;
}
$now = strtotime('7:59pm');
$open = isOpen($now, $times);
if ($open == 0) {
echo "Is closed";
} else {
echo "Is open. Will close in ".ceil($open/60)." minutes";
}
?>
Я провел несколько тестов, и, похоже, он работает как ожидалось, принимая все аспекты, которые я мог бы придумать. сообщите мне, если вы найдете проблему с этим.
Я знаю, что подход выглядит немного неприятным, но я хотел использовать только простые функции (ну, кроме сложной части с array_map
) и держать его как можно короче.
Ответ 7
Если вы используете базу данных для этого, почему вы не использовали datetime для этого.
Пример:
sunday 14:28, saturday 1:28
Вы можете разделить эту две части и сравнить их в течение времени строки (часть 2). Вы можете использовать strtotime, чтобы преобразовать время строки в метку времени и сравнить ее.
Пример:
$date = "sunday 14:28";
echo $stamp = strtotime($date);
Вывод:
1360492200
Как этот код:
$Time="sunday 14:28 , saturday 1:28";
$tA=explode(",",$Time);
$start=strtotime($tA[0]);
$end=strtotime($tA[1]);
$now=time();
if($now>$start and $now<$end){
echo "is open";
}else{
echo "is close";
}
Но у вас есть проблема с их обновлением, вы можете это сделать.
Ответ 8
Вот мое решение:
ВХОДНЫЕ ДАННЫЕ:
$meta = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '9am - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '9am - 3pm',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
current_time ('timestamp') (как сказал автор) аналог time()
в WordPress
И решение:
$now = (int) current_time( 'timestamp' );
$day = strtolower(date('D', $now));
$yesterday = strtolower(date('D', strtotime('-1 day')));
$days = array(
'yesterday' => $meta['opening_hours_'.$yesterday],
'today' => $meta['opening_hours_'.$day],
);
$status = false;
foreach($days as $when=>$times)
{
$parts = explode(',',$times);
foreach($parts as $p)
{
if($p == 'closed')
break;
else{
list($b,$e) = explode('-',$p);
$b = strtotime("$when $b");
$e = strtotime("$when $e");
if($b > $e)
$e += strtotime("$when $e +1 day");;
if($b <= $now && $now <= $e)
{
$status =true;
break;
}
}
}
}
ДЛЯ ТЕСТИРОВАНИЯ:
вы можете изменить первые 3 строки следующим образом:
$now = (int) strtotime('today 3:00am');
$day = strtolower(date('D', $now));
$yesterday = strtolower(date('D', strtotime('yesterday 3:00am')));
Ответ 9
Решение без изменения сохраненного формата времени
<?php
$times = array(
'opening_hours_mon' => '9am - 8pm',
'opening_hours_tue' => '5pm - 2am',
'opening_hours_wed' => '8:30am - 2am',
'opening_hours_thu' => '9am - 3pm',
'opening_hours_fri' => '8:30am - 11am',
'opening_hours_sat' => '9am - 3pm, 5pm - 2am',
'opening_hours_sun' => 'closed'
);
//$time_go = '13:00';
$time_go = date('H:i');
//$day_go = 1; //monday
$day_go = (date('N') - 1);
if(Are_they_open($time_go, $day_go, $times)){
echo 'jep';
}
else{
echo 'nope';
}
function Are_they_open($time_go, $day_go, $times){
// the magic
$times = array_values($times);
$day_go = explode(',', $times[$day_go]);
$time_go = hh_mm_toSeconds($time_go);
foreach($day_go as $time){
if((!$time) || ($time == 'closed')){
return false;
}
$time = explode(' - ', $time);
$time_open = hh_mm_toSeconds(date('H:i', strtotime($time[0])));
$time_close = hh_mm_toSeconds(date('H:i', strtotime($time[1])));
if(($time_go > $time_open) && ($time_go < $time_close)){
return true;
}
elseif (($time_open > $time_close) || ($time_go > $time_open)){
return true;
}
}
return false;
}
function hh_mm_toSeconds($str_time){
sscanf($str_time, "%d:%d", $hours, $minutes);
return ($hours * 3600) + ($minutes * 60);
}
?>
Решение, в котором вы изменяете формат времени
$times = array(
1 => array(
array('07:00', '17:00')
),
2 => array(
array('07:00', '14:30'),
array('15:00', '20:00')
),
3 => array(
array('07:00', '17:00')
),
4 => false, //closed
5 => array(
array('07:00', '17:00'),
array('20:00', '24:00')
),
6 => array(
array('00:00', '03:00'),
array('07:00', '17:00'),
array('20:00', '24:00')
),
7 => array(
array('00:00', '03:00')
),
);
Ответ 10
Передайте свой массив как аргумент этой функции, и вы получите истинное значение для открытого и ложного для закрытого в настоящее время. Это простая простая функция. Я просто проверяю сегодня время открытия и, если необходимо, вчера без лишних циклов в течение всех недельных дней. Возможно, это может быть немного улучшено, но оно работает и не сложно.
function isOpen($times) {
$times = array_values($times); //we will use numeric indexes
$now = new DateTime();
$day = $now->format('N');
$day--; //days are counted 1 to 7 so we decrement it to match indexes
$period = $times[$day];
if($period!='closed') {
$opening = explode('-', $period);
$open = new DateTime($opening[0]);
$close = new DateTime($opening[1]);
if($close<$open) {
//it means today we close after midnight, it is tomorrow
$close->add(new DateInterval('P1D'));
}
if($open<$now && $now<$close) {
//we are open
return true;
}
}
if($period=='closed' || $now<$open) {
//now we check if we still open since yesterday
$day = $day==0 ? 6 : $day-1;
$period = $times[$day];
if($period=='closed') return false;
$opening = explode(' - ', $period);
$open = new DateTime($opening[0]);
$close = new DateTime($opening[1]);
if($close<$open) {
//it means yesterday we closed after midnight
if($now<$close) {
//we are before closing time
return true;
}
}
}
return false;
}
Ответ 11
Возможно, я не понимаю этого полностью, но, пройдя базовый цикл, возможно, это может сработать:
if( ($date('D') == "3" ) )
{
$open = "8";
$close = "17";
$now = time();
if( ($now > $open) and ($now < $close) )
{
echo "open";
}
else
{
echo "closed";
}
}
Возможно, это не лучший способ обойти это, поскольку это не будет включать в себя праздники и т.д. и требует нескольких условных операторов, но я думаю, что это может сработать. Ну, не оптимально, но всегда работал на меня.