37

我需要检查当前时间是否在时间范围内。最简单的情况time_end > time_start:

if time(6,0) <= now.time() <= time(12,00): print '1'

但是当用户进入结束时间小于开始时间的时间范围时,麻烦就开始了,例如“23:00 - 06:00”。像“00:00”这样的时间将在此范围内。大约 5 年前,我编写了这个 PHP 函数:

function checkInterval($start, $end)
  {    
    $dt = date("H:i:s");    

    $tstart = explode(":", $start);
    $tend =   explode(":", $end);
    $tnow =   explode(":", $dt);

    if (!$tstart[2])
      $tstart[2] = 0;

    if (!$tend[2])
      $tend[2] = 0;  

    $tstart = $tstart[0]*60*60 + $tstart[1]*60 + $tstart[2];
    $tend   = $tend[0]*60*60   + $tend[1]*60   + $tend[2];
    $tnow   = $tnow[0]*60*60   + $tnow[1]*60   + $tnow[2];

    if ($tend < $tstart)
      {
        if ($tend - $tnow > 0 && $tnow > $tstart)
          return true;
        else if ($tnow - $tstart > 0 && $tnow > $tend)
          return true;
        else if ($tend > $tnow && $tend < $tstart && $tstart > $tnow)
          return true;
        else return false;
      } else
      {
        if ($tstart < $tnow && $tend > $tnow)
          return true;
        else
          return false;
      }

现在我需要做同样的事情,但我想让它好看。那么,我应该使用什么算法来确定当前时间“00:00”是否在反向范围内,例如['23:00', '01:00']

4

2 回答 2

57

Python 解决方案将变得非常非常短。

def time_in_range(start, end, x):
    """Return true if x is in the range [start, end]"""
    if start <= end:
        return start <= x <= end
    else:
        return start <= x or x <= end

使用、和的datetime.time类。startendx

>>> import datetime
>>> start = datetime.time(23, 0, 0)
>>> end = datetime.time(1, 0, 0)
>>> time_in_range(start, end, datetime.time(23, 30, 0))
True
>>> time_in_range(start, end, datetime.time(12, 30, 0))
False
于 2012-05-25T03:12:35.107 回答
12

日期/时间比你想象的要复杂

涉及日期/时间的计算可能非常棘手,因为您必须考虑时区、闰年、夏令时和许多极端情况。Taavi Burns 在 PyCon2012 的演讲中有一段启发性的视频,题为“你需要了解的关于日期时间的知识”:

您需要了解的关于 datetimes :
time,datetimecalendar标准库的内容有点混乱。找出:在哪里使用什么以及如何使用(特别是当您有许多时区的用户时),以及您可能想要研究哪些额外的模块。

事件:PyCon US 2012 /演讲者:Taavi Burns /录制时间:2012 年 3 月 10 日

使用时区感知日期时间进行计算

明天的概念datetime.time是无效的,因为datetime.time缺少任何日期信息。在比较之前,您可能希望将所有内容都转换为时区感知datetime.datetime

def time_in_range(start, end, x):
    today = timezone.localtime().date()
    start = timezone.make_aware(datetime.datetime.combine(today, start))
    end = timezone.make_aware(datetime.datetime.combine(today, end))
    x = timezone.make_aware(datetime.datetime.combine(today, x))
    if end <= start:
        end += datetime.timedelta(days=1) # tomorrow!
    if x <= start
        x += datetime.timedelta(days=1) # tomorrow!
    return start <= x <= end
于 2012-05-25T03:13:12.090 回答