带有负值的Python timedelta问题

时间:2011-12-06 23:28:17

标签: python timedelta

嗨,我需要一些帮助来理解为什么会这样。 我有一种跟踪事件程序中“剩余时间”的方法:

def get_program_time_budget(self):
    return self.estimated_duration-self.get_program_duration() 

在estimated_duration>时全部罚款self.get_program_duration()但是当事情变得有趣时,事情会变得很有趣。

结果显示给用户:

Estimated   11 hours    Allocated       10 hours 55 minutes     Remaining       5 minutes

当结果为负时,它会这样做:

Estimated   11 hours    Allocated       11 hours 5 minutes  Remaining       -1 day 23 hours 55 minutes

任何想法如何获得结果-5分钟?

编辑: 这是timedelta格式化程序(注意这是一个Django过滤器,因此接收timedelta值作为str - 但它存储为timedelta):

def format_duration(value):
  try:
    delim = ':'
    toks = value.split(',')
    hour = minute = ''
    d_string = value.count('day') and toks[0] or ''
    h, m, s = d_string and toks[-1].strip().split(delim) or value.split(delim)
    try:
        hour = int(h)
    except:
        pass
    try:
        minute = int(m)
    except:
        pass  
    h_string = "%s%s%s" % (hour and hour or '', (hour and ' hour' or ''),(hour and hour > 1 and 's' or '')  )
    m_string = "%s%s%s" % (minute and minute or '', (minute and ' minute' or ''),(minute and minute > 1 and 's' or ''))
    return "%s %s %s" % (d_string, h_string, m_string)
  except Exception, e:
    logging.error("Error in format_duration -> %s. Duration value=%s" % (e, value))
    return ''v 

3 个答案:

答案 0 :(得分:40)

如果您使用的是Python 2.7或更高版本,则可以使用timedelta.total_seconds()将timedelta的float表示形式作为正数或负数秒。

>>> datetime.timedelta(-1, 86100).total_seconds()
-300.0

您应该可以使用它来相当容易地计算分钟数。

如果您不使用Python 2.7,则可以使用文档中的以下等效公式:

(td.microseconds + (td.seconds + td.days * 24 * 3600) * 10**6) / 10.0**6

编辑:看起来您可能正在使用timedelta的默认字符串表示来显示结果,因此我的原始答案可能没那么有用。我会建议这样的东西来显示结果:

def get_program_time_budget(self):
    td = self.estimated_duration-self.get_program_duration()
    if td.days < 0:
        return '-' + str(datetime.timedelta() - td)
    return str(td)

现在返回一个字符串而不是timedelta,对于负时间,它会在正时间delta之前加上' - '。

答案 1 :(得分:13)

为什么?

可能是//%定义方式的意外副作用。

可能是因为它更容易实现datetime类。纪元前的五分钟是23:55,而不是0:-5。

这并不重要。只需知道dayssecondsmicroseconds的归一化方式。而且它很容易解决。

def format_timedelta(td):
    if td < timedelta(0):
        return '-' + format_timedelta(-td)
    else:
        # Change this to format positive timedeltas the way you want
        return str(td)

 >>> format_timedelta(timedelta(minutes=-5))
 '-0:05:00'

答案 2 :(得分:0)

此外,如果您遇到包含负值的 timedelta 对象的问题,即使这些值应该是正值,您也可以使用内置的 python abs(td_object)

>>> current_time = datetime.datetime.now()
>>> fifteen_seconds = datetime.timedelta(seconds=15)
>>> time_delta_after_calculations = current_time - (current_time + fifteen_seconds)  # It should give a timedelta with 15 seconds but it does not
>>> time_delta_after_calculations
datetime.timedelta(days=-1, seconds=86385)
>>> # The above is kind of True but not what expected (A day contains 86400 seconds)
>>> abs(time_delta_after_calculations)  # Gives expected output
datetime.timedelta(seconds=15)