如何在Python中截断DateTime对象上的时间?


251

截断python datetime对象的经典方法是什么?

在这种特殊情况下,直到今天。因此,基本上将小时,分钟,秒和微秒设置为0。

我希望输出也是日期时间对象,而不是字符串。

Answers:


376

我认为这就是您要寻找的...

>>> import datetime
>>> dt = datetime.datetime.now()
>>> dt = dt.replace(hour=0, minute=0, second=0, microsecond=0) # Returns a copy
>>> dt
datetime.datetime(2011, 3, 29, 0, 0)

但是,如果您真的不在乎事物的时间方面,那么您实际上应该只在传递date对象……

>>> d_truncated = datetime.date(dt.year, dt.month, dt.day)
>>> d_truncated
datetime.date(2011, 3, 29)

14
使用时区感知的dt,datetime.datetime(dt.year,dt.month,dt.day)会丢弃tzinfo信息。
ʇsәɹoɈ

1
如果您只是在今天寻找,也可以执行datetime.date.today()
令人赞叹的

4
请注意,python 2和python 3 docs都声明该replace()方法返回一个datetime对象,因此正确的咒语是:dt = datetime.datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)
Brad M

3
OP需要datetime,而不是date对象(您可以使用dt.date()调用(无需使用显式构造函数))。.replace()如果datetime添加纳秒级支持,该方法可能会失败。你可以使用datetime.combine()代替
jfs

@chrisw为什么不将其写成一行datetime.datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)
3kstc '18 -10-4


34

四年后:另一种方式,避免 replace

我知道四年前已经接受的答案有效,但是这似乎比使用replace以下要轻:

dt = datetime.date.today()
dt = datetime.datetime(dt.year, dt.month, dt.day)

笔记

  • 当您在datetime不将时间属性传递给构造函数的情况下创建对象时,将出现午夜。
  • 正如其他人所指出的那样,这假设您想要一个datetime对象供以后与timedelta一起使用。
  • 当然,您可以将其替换为第一行: dt = datetime.datetime.now()

20

您不能截断日期时间对象,因为它是不可变的

但是,这是一种使用0小时,分钟,秒和微秒字段构造新日期时间而不丢弃原始日期或tzinfo的方法:

newdatetime = now.replace(hour=0, minute=0, second=0, microsecond=0)

1
+1:如果您将replace期权放在第一位,因为这可能就是他们想要的。
S.Lott

使用不正确tzinfo=now.tzinfo。在tzinfo半夜可能会有所不同例如,UTC偏移量2012-04-01 00:09:00在澳大利亚/墨尔本时区(上午9:00)是AEST+10:00,但它是AEDT+11:002012-04-01 00:00:00(午夜) -有在这一天结束DST过渡。您可以使用pytz模块来修复它,请参阅我的答案。
jfs

13

要获得对应于给定日期时间对象的午夜,可以使用datetime.combine()method

>>> from datetime import datetime, time
>>> dt = datetime.utcnow()
>>> dt.date()
datetime.date(2015, 2, 3)
>>> datetime.combine(dt, time.min)
datetime.datetime(2015, 2, 3, 0, 0)

.replace()方法相比,优点是datetime.combine()即使datetime模块引入了纳秒级支持,基于解决方案的解决方案仍将继续有效。

tzinfo可以在必要时保留它,但是utc偏移量可能在午夜有所不同,例如由于DST转换,因此幼稚的解决方案(设置tzinfo时间属性)可能会失败。请参阅如何获取给定时区的UTC时间“午夜”?


7

您可以为此使用熊猫(尽管对于该任务可能是开销)。您可以像通常的数字一样使用roundfloorceil,也可以使用offset-aliases的任何熊猫频率:

import pandas as pd
import datetime as dt

now = dt.datetime.now()
pd_now = pd.Timestamp(now)

freq = '1d'
pd_round = pd_now.round(freq)
dt_round = pd_round.to_pydatetime()

print(now)
print(dt_round)

"""
2018-06-15 09:33:44.102292
2018-06-15 00:00:00
"""

4

您可以使用datetime.strftime提取日期,月份,年份...

范例:

from datetime import datetime
d = datetime.today()

# Retrieves the day and the year
print d.strftime("%d-%Y")

输出(今天):

29-2011

如果只想获取日期,则可以使用day属性,例如:

from datetime import datetime
d = datetime.today()

# Retrieves the day
print d.day

Ouput(今天):

29

好吧,事情是我已经做过一次,这样可能会比在datetime对象中设置小时最小值等字段产生更多的开销。
凯尔·勃兰特

嘿,那是一个奇怪的方式做到这一点,你实际上可以只是做d.day等等
约亨Ritzel

3

有一个很棒的用于操作日期的库:Delorean

import datetime
from delorean import Delorean
now = datetime.datetime.now()
d = Delorean(now, timezone='US/Pacific')

>>> now    
datetime.datetime(2015, 3, 26, 19, 46, 40, 525703)

>>> d.truncate('second')
Delorean(datetime=2015-03-26 19:46:40-07:00, timezone='US/Pacific')

>>> d.truncate('minute')
Delorean(datetime=2015-03-26 19:46:00-07:00, timezone='US/Pacific')

>>> d.truncate('hour')
Delorean(datetime=2015-03-26 19:00:00-07:00, timezone='US/Pacific')

>>> d.truncate('day')
Delorean(datetime=2015-03-26 00:00:00-07:00, timezone='US/Pacific')

>>> d.truncate('month')
Delorean(datetime=2015-03-01 00:00:00-07:00, timezone='US/Pacific')

>>> d.truncate('year')
Delorean(datetime=2015-01-01 00:00:00-07:00, timezone='US/Pacific')

并且如果您想获取日期时间值:

>>> d.truncate('year').datetime
datetime.datetime(2015, 1, 1, 0, 0, tzinfo=<DstTzInfo 'US/Pacific' PDT-1 day, 17:00:00 DST>)

如果结果时间具有不同的utc偏移,例如由于DST转换,则返回错误的时间(错误的utc偏移)。请参阅如何获取给定时区的UTC时间“午夜”?
jfs


1

有一个模块datetime_truncate可以为您处理。它只是调用datetime.replace。


1

6年后...我发现了这篇文章,并且我更喜欢numpy的方式:

import numpy as np
dates_array = np.array(['2013-01-01', '2013-01-15', '2013-01-30']).astype('datetime64[ns]')
truncated_dates = dates_array.astype('datetime64[D]')

干杯


1
>>> import datetime
>>> dt = datetime.datetime.now()
>>> datetime.datetime.date(dt)
datetime.date(2019, 4, 2)

1

你可以用

datetime.date.today()

轻巧,可精确返回您想要的东西。


之前已经给出了相同的答案,没有新内容。
slfan

抱歉@slfan,但即使使用浏览器中的“搜索”,我也看不到任何带有您名字的帖子。
Bordotti

1
不是我,而是3年前的zx81。搜索datetime.date.today()。如果答案正确,则应该投票,而不要再回答。
slfan


0

如果要处理DateTime类型的Series,则有一种更有效的方法来截断它们,尤其是在Series对象包含很多行时。

您可以使用发言权功能

例如,如果要将其截断为几个小时:

产生日期范围

times = pd.Series(pd.date_range(start='1/1/2018 04:00:00', end='1/1/2018 22:00:00', freq='s'))

我们可以通过比较替换功能和发言权功能之间的运行时间来检查它。

%timeit times.apply(lambda x : x.replace(minute=0, second=0, microsecond=0))
>>> 341 ms ± 18.2 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)

%timeit times.dt.floor('h')
>>>>2.26 ms ± 451 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)

0

这是适合另一行但又不是特别优雅的另一种方式:

dt = datetime.datetime.fromordinal(datetime.date.today().toordinal())
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.