编程入门16:Python时间操作

上一篇:编程入门15:Python迭代机制

Python标准库包含有一个time模块用于基本的时间处理,其中的time()函数会读取系统时钟并返回float类型的Unix纪元“时间戳”(Timestamp),即当前时间距离国际标准时间1970年1月1日0点的秒数。另一个sleep()函数则会让程序“休眠”指定的秒数再继续执行:

In [1]: import time

In [2]: time.time()  # 北京时间2018年8月15日21时48分
Out[2]: 1534340898.5661483

In [3]: for i in range(1, 11):
   ...:     time.sleep(1)
   ...:     print("{}秒".format(i), end=" ")
   ...:     
1秒 2秒 3秒 4秒 5秒 6秒 7秒 8秒 9秒 10秒 

如果你想要确定一段程序的运行时间,可以在代码块的首尾分别获取时间并相减——比较相对时间推荐使用以下两个函数更为精确高效:perf_counter()返回当前系统开机运行的秒数,process_time()则返回当前进程实际运行的秒数(休眠时间不算在内)。

In [4]: t = time.time()
   ...: time.sleep(5)
   ...: print(time.time()-t)
5.002017259597778

In [5]: t = time.perf_counter()
   ...: time.sleep(5)
   ...: print(time.perf_counter()-t)
5.000831686000026

In [6]: t = time.process_time()
   ...: time.sleep(5)
   ...: print(time.process_time()-t)
0.0

如果你想基于公历来进行日期时间的显示和运算,就要引入datetime模块,该模块主要包含datetime类型,其中的类方法fromtimestamp()或utcfromtimestamp()可以将时间戳转换为表示对应地方时或国际标准时的datetime对象:

In [7]: import datetime as dt

In [8]: t = time.time()

In [9]: dt.datetime.fromtimestamp(t)  # 地方时(北京时间)
Out[9]: datetime.datetime(2018, 8, 15, 21, 49, 57, 47818)

In [10]: dt.datetime.utcfromtimestamp(t)  # 国际标准时(格林威治时间)
Out[10]: datetime.datetime(2018, 8, 15, 13, 49, 57, 47818)

datetime对象包含int类型的year、month、day、hour、minute、second和microsecond属性,对应年、月、日、时、分、秒和微秒。你可以调用datetime类构造器返回datetime对象,也可以调用类方法now()或utcnow()返回表示当前地方时或国际标准时的datetime对象。

datetime模块还提供了timedelta类型来表示时间差,timedelta类的构造器接受关键字参数weeks、days、hours、minutes、seconds、milliseconds和microseconds指定时间差是多少周、日、时、分、秒、毫秒和微秒——没有年和月因其长度不固定。timedelta对象可以与datetime对象做加减,也可以彼此相加减,还可以乘以或除以int或float数值。

In [11]: d = dt.datetime.now()

In [12]: str(d)  # datetime转字符串,返回标准日期格式
Out[12]: '2018-08-15 21:50:51.605667'

In [13]: d1000 = d + dt.timedelta(days=1000)  # 1000天之后的日期

In [14]: print(d1000)
2021-05-11 21:50:51.605667

datetime模块还包含timezone类型来表示时区,datetime对象要加上timezone类型的tzinfo属性才能无歧义地定位时间点。timezone类构造器的参数是表示与零时区时差的timedelta对象,你也可以引用time.timezone变量从系统时钟得到国际标准时减地方时的秒数,该值取反就是当前时区的时差。

In [15]: d0 = dt.datetime(1970, 1, 1, 0, tzinfo=dt.timezone.utc)  # Unix纪元零时的datetime

In [16]: d0
Out[16]: datetime.datetime(1970, 1, 1, 0, 0, tzinfo=datetime.timezone.utc)

In [17]: print(d0)  # Unix纪元零时的标准时间格式
1970-01-01 00:00:00+00:00

In [18]: d0.timestamp()  # Unix纪元零时的时间戳
Out[18]: 0.0

In [19]: d = dt.datetime.now()  # 当前本地时的datetime,不包含时区信息

In [20]: d
Out[20]: datetime.datetime(2018, 8, 15, 21, 53, 7, 74861)

In [21]: d = d.replace(tzinfo=dt.timezone(dt.timedelta(seconds=-time.timezone)))  # 加上时区信息,从系统时钟获取

In [22]: print(d)  # 包含时区的标准时间格式
2018-08-15 21:53:07.074861+08:00

In [23]: d.astimezone(dt.timezone(dt.timedelta(hours=9)))  # 转换为东九区时
Out[23]: datetime.datetime(2018, 8, 15, 22, 53, 7, 74861, tzinfo=datetime.timezone(datetime.timedelta(seconds=32400)))

你还可以用datetime实例的strftime()方法返回指定格式的日期字符串,或用strptime()类方法将日期字符串转为datetime对象——日期格式字符串中使用百分号打头的占位符来包含日期信息,这是继承自C语言的标准格式,详情可以参看官方文档 https://docs.python.org/3/library/datetime.html#strftime-and-strptime-behavior

In [24]: d.strftime("%Y年%m月%d日%H时%M分%S秒".encode("unicode-escape").decode()).encode().decode("unicode-escape")
Out[24]: '2018年08月15日21时53分07秒'

In [25]: dt.datetime.strptime("1970年1月1日0时", "%Y年%m月%d日%H时")
Out[25]: datetime.datetime(1970, 1, 1, 0, 0)

下面再来看一个定时监测网站状态的程序:

"""xmonitor.py 网站定时监测
定时访问CIV论坛,新增帖子数大于5或发生网络异常则发邮件示警
"""
from urllib.request import urlopen, Request
import re
import time
import smtplib
from email.mime.text import MIMEText
from email.header import Header
from email.utils import formataddr
mail_sender = "******@sohu.com"
mail_receiver = "******@163.com"
smtp_host = "smtp.sohu.com"  # 发送邮件配置,请参看邮箱帮助文档开通SMTP服务
smtp_user = "******@sohu.com"
smtp_pass = "******"
site = "CIV论坛"
url = "http://www.civclub.net/bbs"  # 这是个策略游戏论坛,欢迎大家访问!
headers = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:61.0) \
           Gecko/20100101 Firefox/61.0"}


def mail(title, content):
    """发送邮件
    """
    msg = MIMEText(content, "plain", "utf-8")
    msg["From"] = formataddr(["监测员", mail_sender])
    msg["To"] = formataddr(["管理员", mail_receiver])
    msg["Subject"] = Header(f"{site}:{title}", "utf-8")
    try:
        smtp = smtplib.SMTP(smtp_host)
        smtp.ehlo()
        smtp.starttls()
        smtp.login(smtp_user, smtp_pass)
        smtp.sendmail(mail_sender, mail_receiver, msg.as_string())
        print("邮件发送成功")
    except Exception as e:
        print(f"邮件发送失败:{repr(e)}")


def main():
    """每小时获取新增帖数,大于5或发生网络异常则发邮件示警
    """
    posts = 0  # 发帖数
    while True:
        print(f"检查网站:{url}")
        try:
            req = Request(url=url, headers=headers)
            html = urlopen(req).read().decode("gbk")
            match = re.search(r'今日: (\d+?)', html)
            newposts = int(match.group(1))
            increase = newposts - posts
            if increase > 5:
                mail("发帖量大", f"近一小时新增{increase}个帖子")
            posts = newposts
        except Exception as e:
            mail("异常错误", repr(e))
        print("进入休眠……")
        for _ in range(3600):  # 休眠一小时
            time.sleep(1)


if __name__ == "__main__":
    main()
编程入门16:Python时间操作_第1张图片
16_mail.png

上面的代码用3600次sleep(1)而非sleep(3600)来实现休眠一小时,以便能随时按Ctrl+C中断程序(如果是用Spyder则在IPython面板中右击选择Quit结束运行)。

——编程原来是这样……

编程小提示:字符串格式化

Python有多种字符串格式化语法,第一种是在字符串内加百分号占位符,这是继承自C语言的标准格式,详情可参看 https://docs.python.org/3/library/stdtypes.html#printf-style-string-formatting
第二种是str.format()方法,在字符串内加花括号占位符,这是目前普遍使用的语法,详情可参看 https://docs.python.org/3/library/stdtypes.html#str.format
第三种就是Python 3.6新增的“格式字符串”(f-strings),在带f前缀的字符串内加花括号,其中可以放任意表达式,这种语法最为简洁自然,如果你的程序不需要兼容旧版本,那么推荐用格式字符串,详情可参看 https://docs.python.org/3/reference/lexical_analysis.html#f-strings

In [26]: name, year = "Python", 1990

In [27]: "%s语言诞生于%s年。" % (name, year)
Out[27]: 'Python语言诞生于1990年。'

In [28]: "{}语言诞生于{}年。".format(name, year)
Out[28]: 'Python语言诞生于1990年。'

In [29]: f"{name}语言诞生于{year}年。"
Out[29]: 'Python语言诞生于1990年。'

你可能感兴趣的:(编程入门16:Python时间操作)