44

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

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值作为 a str- 但它存储为 a 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 
4

3 回答 3

66

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

>>> 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,对于负 timedelta,它会在正 timedelta 前面加上一个“-”。

于 2011-12-06T23:45:52.170 回答
26

为什么?

可能作为一种意想不到的副作用的方式//%被定义。

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

这并不重要。只知道它是如何days, seconds, 和microseconds标准化的。而且它可以很容易地解决。

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'
于 2011-12-07T00:36:31.823 回答
0

此外,如果您遇到timedelta包含负值的对象的问题,即使值应该是正的,您也可以使用内置的 pythonsabs(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)
于 2021-07-03T16:18:42.943 回答