每周一个 Python 模块 | datetime

datetime 包含用于处理日期和时间的函数和类。

Time

时间值用time类表示。 time实例有属性hourminutesecond,和microsecond,还可以包括时区信息。

import datetime

t = datetime.time(1, 2, 3)
print(t)    # 01:02:03
print('hour       :', t.hour)   # hour       : 1
print('minute     :', t.minute) # minute     : 2
print('second     :', t.second) # second     : 3
print('microsecond:', t.microsecond)    # microsecond: 0
print('tzinfo     :', t.tzinfo) # tzinfo     : None

time实例只保留时间值,而不是与时间相关联的日期。

import datetime

print('Earliest  :', datetime.time.min) # Earliest  : 00:00:00
print('Latest    :', datetime.time.max) # Latest    : 23:59:59.999999
print('Resolution:', datetime.time.resolution)  # Resolution: 0:00:00.000001

minmax类属性反映了时间在一天的有效范围。

时间的分辨率限制在整个微秒。(也可以说是一个微秒,详见上面代码 Resolution)

import datetime

for m in [1, 0, 0.1, 0.6]:
    try:
        print('{:02.1f} :'.format(m),
              datetime.time(0, 0, 0, microsecond=m))
    except TypeError as err:
        print('ERROR:', err)
  
# 输出
# 1.0 : 00:00:00.000001
# 0.0 : 00:00:00
# ERROR: integer argument expected, got float
# ERROR: integer argument expected, got float        

微秒的浮点值会导致TypeError

Date

日期值用date 类表示。实例具有属性yearmonthday。使用today()类方法可以轻松创建当前日期。

import datetime

today = datetime.date.today()
print(today)    # 2018-03-18
print('ctime  :', today.ctime())    # ctime  : Sun Mar 18 00:00:00 2018
tt = today.timetuple()
print('tuple  : tm_year  =', tt.tm_year)    # tuple  : tm_year  = 2018
print('         tm_mon   =', tt.tm_mon)     # tm_mon   = 3
print('         tm_mday  =', tt.tm_mday)    # tm_mday  = 18
print('         tm_hour  =', tt.tm_hour)    # tm_hour  = 0
print('         tm_min   =', tt.tm_min)     # tm_min   = 0
print('         tm_sec   =', tt.tm_sec)     # tm_sec   = 0
print('         tm_wday  =', tt.tm_wday)    # tm_wday  = 6
print('         tm_yday  =', tt.tm_yday)    # tm_yday  = 77
print('         tm_isdst =', tt.tm_isdst)   # tm_isdst = -1
print('ordinal:', today.toordinal())    # ordinal: 736771
print('Year   :', today.year)   # Year   : 2018
print('Mon    :', today.month)  # Mon    : 3
print('Day    :', today.day)    # Day    : 18

还有一些类方法,用于从 POSIX 时间戳或整数表示公历中的日期值创建实例,其中 1 年 1 月 1 日为 1,每个后续日期将值递增 1。

import datetime
import time

o = 733114
print(o)    # o               : 733114
print(datetime.date.fromordinal(o)) # fromordinal(o)  : 2008-03-13

t = time.time()
print(t)    # t               : 1521404434.262209
print(datetime.date.fromtimestamp(t))   # fromtimestamp(t): 2018-03-18

此示例说明了fromordinal()fromtimestamp() 使用的不同值类型。

time一样,可以使用minmax属性确定支持的日期值范围。

import datetime

print('Earliest  :', datetime.date.min) # Earliest  : 0001-01-01
print('Latest    :', datetime.date.max) # Latest    : 9999-12-31
print('Resolution:', datetime.date.resolution)  # Resolution: 1 day, 0:00:00

日期的精确度是整天。

创建新date实例的另一种方法是使用replace()

import datetime

d1 = datetime.date(2008, 3, 29)
print('d1:', d1.ctime())    # d1: Sat Mar 29 00:00:00 2008

d2 = d1.replace(year=2009)
print('d2:', d2.ctime())    # d2: Sun Mar 29 00:00:00 2009

此示例更改年份,日期和月份保持不变。

timedeltas

可以使用两个datetime对象的做基本的计算,或通过将 datetimetimedelta 组合来计算未来和过去的日期。减去日期会产生一个timedelta,也可以从日期中添加或减去timedelta以产生另一个日期。timedelta的内部值以天,秒和微秒存储。

import datetime

print('microseconds:', datetime.timedelta(microseconds=1))  # 0:00:00.000001
print('milliseconds:', datetime.timedelta(milliseconds=1))  # 0:00:00.001000
print('seconds     :', datetime.timedelta(seconds=1))   # 0:00:01
print('minutes     :', datetime.timedelta(minutes=1))   # 0:01:00
print('hours       :', datetime.timedelta(hours=1)) # 1:00:00
print('days        :', datetime.timedelta(days=1))  # 1 day, 0:00:00
print('weeks       :', datetime.timedelta(weeks=1)) # 7 days, 0:00:00

传递给构造函数的中间级别值将转换为天,秒和微秒。

timedelta的完整持续时间可以使用total_seconds()以秒数来检索。

import datetime

for delta in [datetime.timedelta(microseconds=1),
              datetime.timedelta(milliseconds=1),
              datetime.timedelta(seconds=1),
              datetime.timedelta(minutes=1),
              datetime.timedelta(hours=1),
              datetime.timedelta(days=1),
              datetime.timedelta(weeks=1),
              ]:
    print('{:15} = {:8} seconds'.format(
        str(delta), delta.total_seconds())
    )

# 输出    
# 0:00:00.000001  =    1e-06 seconds
# 0:00:00.001000  =    0.001 seconds
# 0:00:01         =      1.0 seconds
# 0:01:00         =     60.0 seconds
# 1:00:00         =   3600.0 seconds
# 1 day, 0:00:00  =  86400.0 seconds
# 7 days, 0:00:00 = 604800.0 seconds    

返回值是一个浮点数,以适应次秒持续时间。

日期计算

日期计算使用标准算术运算符。

import datetime

today = datetime.date.today()
print('Today    :', today)  # Today    : 2018-03-18

one_day = datetime.timedelta(days=1)
print('One day  :', one_day)    # One day  : 1 day, 0:00:00

yesterday = today - one_day
print('Yesterday:', yesterday)  # Yesterday: 2018-03-17

tomorrow = today + one_day
print('Tomorrow :', tomorrow)   # Tomorrow : 2018-03-19

print('tomorrow - yesterday:', tomorrow - yesterday)    # 2 days, 0:00:00
print('yesterday - tomorrow:', yesterday - tomorrow)    # -2 days, 0:00:00

示例说明了使用timedelta 对象计算新日期,可以减去日期实例以生成 timedeltas(包括负 delta 值)。

timedelta对象还支持与整数,浮点数和其他timedelta对象进行算术运算。

import datetime

one_day = datetime.timedelta(days=1)
print('1 day    :', one_day)    # 1 day    : 1 day, 0:00:00
print('5 days   :', one_day * 5)    # 5 days   : 5 days, 0:00:00
print('1.5 days :', one_day * 1.5)  # 1.5 days : 1 day, 12:00:00
print('1/4 day  :', one_day / 4)    # 1/4 day  : 6:00:00

# assume an hour for lunch
work_day = datetime.timedelta(hours=7)
meeting_length = datetime.timedelta(hours=1)
print('meetings per day :', work_day / meeting_length)  # meetings per day : 7.0

在此示例中,计算一天的几个倍数,结果timedelta保持天数或小时数。最后一个示例演示了如何通过组合两个timedelta对象来计算值。在这种情况下,结果是浮点数。

日期和时间比较

可以使用标准比较运算符比较日期和时间值,以确定哪个更早或更晚。

import datetime
import time

print('Times:') # Times:
t1 = datetime.time(12, 55, 0)
print('  t1:', t1)  # t1: 12:55:00
t2 = datetime.time(13, 5, 0)
print('  t2:', t2)  # t2: 13:05:00
print('  t1 < t2:', t1 < t2)    # t1 < t2: True

print('Dates:') # Dates:
d1 = datetime.date.today()
print('  d1:', d1)  # d1: 2018-03-18
d2 = datetime.date.today() + datetime.timedelta(days=1)
print('  d2:', d2)  # d2: 2018-03-19
print('  d1 > d2:', d1 > d2)    # d1 > d2: False

支持所有比较运算符。

组合日期和时间

使用datetime类来保存由日期和时间组件组成的值。与date一样,有几个简单的类方法可以创建datetime实例。

import datetime

print('Now    :', datetime.datetime.now())
print('Today  :', datetime.datetime.today())
print('UTC Now:', datetime.datetime.utcnow())
print()

FIELDS = [
    'year', 'month', 'day',
    'hour', 'minute', 'second',
    'microsecond',
]

d = datetime.datetime.now()
for attr in FIELDS:
    print('{:15}: {}'.format(attr, getattr(d, attr)))
    
# 输出
# Now    : 2018-03-18 16:20:34.811583
# Today  : 2018-03-18 16:20:34.811616
# UTC Now: 2018-03-18 20:20:34.811627

# year           : 2018
# month          : 3
# day            : 18
# hour           : 16
# minute         : 20
# second         : 34
# microsecond    : 811817

datetime实例具有datetime对象的所有属性。

date一样,datetime为创建新实例提供了方便的类方法。它还包括 fromordinal()fromtimestamp()

import datetime

t = datetime.time(1, 2, 3)
print('t :', t)     # t : 01:02:03

d = datetime.date.today()
print('d :', d)     # d : 2018-03-18

dt = datetime.datetime.combine(d, t)
print('dt:', dt)    # dt: 2018-03-18 01:02:03

combine()从一个 date和一个time实例创建datetime实例。

格式化和解析

datetime 对象的默认字符串表示形式使用 ISO-8601 格式(YYYY-MM-DDTHH:MM:SS.mmmmmm)。可以使用 strftime() 对其进行格式化。

import datetime

format = "%a %b %d %H:%M:%S %Y"

today = datetime.datetime.today()
print('ISO     :', today)   # ISO     : 2018-03-18 16:20:34.941204

s = today.strftime(format)
print('strftime:', s)   # strftime: Sun Mar 18 16:20:34 2018

d = datetime.datetime.strptime(s, format)
print('strptime:', d.strftime(format))  # strptime: Sun Mar 18 16:20:34 2018

使用datetime.strptime()于格式化字符串转换为 datetime实例。

Python的字符串格式化迷你语言可以使用相同的格式代码,方法是:在格式字符串的字段规范中放置它们。

import datetime

today = datetime.datetime.today()
print('ISO     :', today)   # ISO     : 2018-03-18 16:20:35.006116
print('{:%a %b %d %H:%M:%S %Y}'.format(today))  # Sun Mar 18 16:20:35 2018

每个日期时间格式代码仍必须以前缀为前缀%,后续冒号将作为文字字符处理,以包含在输出中。

下表显示了美国/东部时区 2016 年 1 月 13 日下午 5:00 的所有格式代码。

符号 含义
%a 缩写的工作日名称 'Wed'
%A 完整的工作日名称 'Wednesday'
%w 工作日编号 - 0(星期日)至6(星期六) '3'
%d 每月的一天(零填充) '13'
%b 缩写的月份名称 'Jan'
%B 全月名称 'January'
%m 一年中的一个月 '01'
%y 没有世纪的一年 '16'
%Y 与世纪的一年 '2016'
%H 24小时制的小时 '17'
%I 12小时制的小时 '05'
%p 上午下午 'PM'
%M 分钟 '00'
%S '00'
%f 微秒 '000000'
%z 时区感知对象的UTC偏移量 '-0500'
%Z 时区名称 'EST'
%j 一年中的某一天 '013'
%W 一年中的一周 '02'
%c 当前区域设置的日期和时间表示形式 'Wed Jan 13 17:00:00 2016'
%x 当前区域设置的日期表示形式 '01/13/16'
%X 当前区域设置的时间表示 '17:00:00'
%% 文字%字符 '%'

时区

datetime中,时区由子类tzinfo表示。由于tzinfo是一个抽象基类,应用程序需要定义一个子类并为一些方法提供适当的实现以使其有用。

datetime确实在类timezone中使用了一个有点天真的实现,它使用 UTC 的固定偏移量,并且不支持一年中不同日期的不同偏移值,例如夏令时适用的地方,或者 UTC 的偏移量随时间变化的情况。

import datetime

min6 = datetime.timezone(datetime.timedelta(hours=-6))
plus6 = datetime.timezone(datetime.timedelta(hours=6))
d = datetime.datetime.now(min6)

print(min6, ':', d) 
print(datetime.timezone.utc, ':',
      d.astimezone(datetime.timezone.utc))
print(plus6, ':', d.astimezone(plus6))

# convert to the current system timezone
d_system = d.astimezone()
print(d_system.tzinfo, '      :', d_system)

# UTC-06:00 : 2018-03-18 14:20:35.123594-06:00
# UTC : 2018-03-18 20:20:35.123594+00:00
# UTC+06:00 : 2018-03-19 02:20:35.123594+06:00
# EDT       : 2018-03-18 16:20:35.123594-04:00

要将日期时间值从一个时区转换为另一个时区,请使用 astimezone()。在上面的示例中,显示了 UTC 两侧 6 小时的两个独立时区,并且utc实例 from datetime.timezone也用于参考。最终输出行显示系统时区中的值,通过不带参数的astimezone()调用获取 。

常用操作总结

最后,总结一些常用的函数,可以直接选取自己需要的进行使用,具体代码见 Github 时间模块常用操作



相关文档:

https://pymotw.com/3/datetime/index.html

https://zhuanlan.zhihu.com/p/28209870

http://flowsnow.net/2017/09/07/Python%E6%97%B6%E9%97%B4%E6%A8%A1%E5%9D%97%E5%B8%B8%E7%94%A8%E6%93%8D%E4%BD%9C%E6%80%BB%E7%BB%93/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容

  • 原文链接:http://www.cnblogs.com/lhj588/archive/2012/04/23/246...
    qtruip阅读 1,325评论 0 0
  • 原文:http://www.ahlinux.com/python/12181.html 1 datetime模块定...
    peimin阅读 10,631评论 0 1
  • Datetime模块探索 写在前面 本文介绍了Datetime,常用的内容 naive datetime和 awa...
    胃痛的香蕉1阅读 4,237评论 0 6
  • Python提供了多个内置模块用于操作日期时间,像calendar,time,datetime。time模块我在之...
    晓函阅读 1,382评论 0 0
  • 写一封小情书,写给你,也是写给我自己。 第一次见你时,你2岁,我1岁。你说,我当时被我老妈抱着,小小的手在冲你使...
    邦妮大神阅读 214评论 0 0