在Python中将N秒添加到datetime.time的标准方法是什么?

Pau*_*son 333 python math time datetime

给定datetime.timePython中的值,是否有一种标准方法可以为其添加整数秒,例如11:34:59+ 3 = 11:35:02

这些明显的想法不起作用:

>>> datetime.time(11, 34, 59) + 3
TypeError: unsupported operand type(s) for +: 'datetime.time' and 'int'
>>> datetime.time(11, 34, 59) + datetime.timedelta(0, 3)
TypeError: unsupported operand type(s) for +: 'datetime.time' and 'datetime.timedelta'
>>> datetime.time(11, 34, 59) + datetime.time(0, 0, 3)
TypeError: unsupported operand type(s) for +: 'datetime.time' and 'datetime.time'
Run Code Online (Sandbox Code Playgroud)

最后我写了这样的函数:

def add_secs_to_time(timeval, secs_to_add):
    secs = timeval.hour * 3600 + timeval.minute * 60 + timeval.second
    secs += secs_to_add
    return datetime.time(secs // 3600, (secs % 3600) // 60, secs % 60)
Run Code Online (Sandbox Code Playgroud)

我不禁想到我错过了一个更简单的方法来做到这一点.

有关

pax*_*blo 463

您可以使用完整datetime变量timedelta,并提供虚拟日期,然后使用time以获取时间值.

例如:

import datetime
a = datetime.datetime(100,1,1,11,34,59)
b = a + datetime.timedelta(0,3) # days, seconds, then other fields.
print a.time()
print b.time()
Run Code Online (Sandbox Code Playgroud)

导致两个值,相隔三秒:

11:34:59
11:35:02
Run Code Online (Sandbox Code Playgroud)

您也可以选择更具可读性

b = a + datetime.timedelta(seconds=3)
Run Code Online (Sandbox Code Playgroud)

如果你这么倾向


如果您正在使用可以执行此操作的功能,则可以查看addSecs以下内容:

import datetime

def addSecs(tm, secs):
    fulldate = datetime.datetime(100, 1, 1, tm.hour, tm.minute, tm.second)
    fulldate = fulldate + datetime.timedelta(seconds=secs)
    return fulldate.time()

a = datetime.datetime.now().time()
b = addSecs(a, 300)
print a
print b
Run Code Online (Sandbox Code Playgroud)

这输出:

 09:11:55.775695
 09:16:55
Run Code Online (Sandbox Code Playgroud)

  • 为避免OverflowErrors,我建议使用不同的虚拟日期,例如几年后:datetime(101,1,1,11,34,59).如果你尝试从上面的日期减去一个大的timedelta,你会得到一个"OverflowError:date value out of range"错误,因为datetime对象的年份不能小于1 (6认同)
  • @pheelicks完成了,尽管响应时间有点晚,但响应时间并不完全敏捷:-)由于我不得不修复代码中的另一个错误,因此我认为我会同时采纳您的建议。 (2认同)

Eli*_*ght 48

正如其他人所说,你可以在整个日期时使用完整的日期时间对象:

sometime = get_some_time() # the time to which you want to add 3 seconds
later = (datetime.combine(date.today(), sometime) + timedelta(seconds=3)).time()
Run Code Online (Sandbox Code Playgroud)

但是,我认为值得解释为什么需要完整的日期时间对象.考虑如果我在晚上11点加入2小时会发生什么.什么是正确的行为?一个例外,因为你的时间不能超过晚上11:59?它应该包裹回来吗?

不同的程序员会期望不同的东西,所以他们选择的结果会让很多人感到惊讶.更糟糕的是,程序员会编写在最初测试时运行良好的代码,然后通过做一些意想不到的事情让它破裂.这非常糟糕,这就是为什么不允许将timedelta对象添加到时间对象的原因.

  • 显然,这是一个例外。无法将 *TIME*delta 添加到 *TIME* 是愚蠢的。 (2认同)
  • @JürgenA.Erhard 有趣的是,我“显然”认为正确的行为应该是环绕...... (2认同)

unm*_*ted 19

一件小事,可能会增加清晰度以覆盖秒的默认值

>>> b = a + datetime.timedelta(seconds=3000)
>>> b
datetime.datetime(1, 1, 1, 12, 24, 59)
Run Code Online (Sandbox Code Playgroud)

  • 我喜欢这一个.使用指定的参数很好并且清晰. (4认同)

Pau*_*son 11

感谢@Pax Diablo,@ bvmou和@Arachnid提出的使用完整日期时间的建议.如果我必须接受来自外部源的datetime.time对象,那么这似乎是一个替代add_secs_to_time()函数:

def add_secs_to_time(timeval, secs_to_add):
    dummy_date = datetime.date(1, 1, 1)
    full_datetime = datetime.datetime.combine(dummy_date, timeval)
    added_datetime = full_datetime + datetime.timedelta(seconds=secs_to_add)
    return added_datetime.time()
Run Code Online (Sandbox Code Playgroud)

这个详细的代码可以压缩到这个单行代码:

(datetime.datetime.combine(datetime.date(1, 1, 1), timeval) + datetime.timedelta(seconds=secs_to_add)).time()
Run Code Online (Sandbox Code Playgroud)

但是我想我还是想把它包含在代码清晰度的函数中.


res*_*dsk 8

如果值得为你的项目添加另一个文件/依赖项,我只写了一个小的类,它扩展datetime.time了算术的能力.当你经过午夜时,它会绕零.现在,"现在是什么时间,24小时后"有很多极端情况,包括夏令时,闰秒,历史时区变化等等.但有时你确实需要这个简单的案例,而这就是它的作用.

你的例子将写成:

>>> import datetime
>>> import nptime
>>> nptime.nptime(11, 34, 59) + datetime.timedelta(0, 3)
nptime(11, 35, 2)
Run Code Online (Sandbox Code Playgroud)

nptime继承自datetime.time,所以任何这些方法也应该可用.

它可以从PyPi获得nptime("非迂腐时间"),或者在GitHub上:https://github.com/tgs/nptime


Ven*_*nga 6

在现实环境中,单独使用永远不是一个好主意time,总是使用datetime,甚至更好utc,以避免诸如夜间、夏令时、用户和服务器之间的不同时区等冲突。

所以我推荐这种方法:

import datetime as dt

_now = dt.datetime.now()  # or dt.datetime.now(dt.timezone.utc)
_in_5_sec = _now + dt.timedelta(seconds=5)

# get '14:39:57':
_in_5_sec.strftime('%H:%M:%S')
Run Code Online (Sandbox Code Playgroud)


Bar*_*oon 5

为了完整起见,这里是这样做的方法arrow(Python 的更好的日期和时间):

sometime = arrow.now()
abitlater = sometime.shift(seconds=3)
Run Code Online (Sandbox Code Playgroud)


小智 5

您不能简单地添加数字,datetime因为不清楚使用的单位是秒,小时,周...

timedelta用于日期和时间操作的类。datetime减去datetimeGives timedeltadatetimePlus timedeltaGives datetimedatetime虽然两个对象可以添加,但不能添加两个对象timedelta

创建timedelta要添加多少秒的datetime对象并将其添加到对象:

>>> from datetime import datetime, timedelta
>>> t = datetime.now() + timedelta(seconds=3000)
>>> print(t)
datetime.datetime(2018, 1, 17, 21, 47, 13, 90244)
Run Code Online (Sandbox Code Playgroud)

C ++中有相同的概念:std::chrono::duration

  • 请务必添加说明,新来者可能不知道您在这里做什么。 (3认同)