整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:

HTML网络编程之时间格式

络编程之时间格式。

同学们好,今天我们分享的是如何让搜索引擎等程序更容易地提取网页中的时间信息。我们将使用time标签来实现这一目标。这个标签你们可能已经有所了解,但是现在不需要掌握太多细节,只需要知道它的作用即可。

现在来看看我们的示例页面,可以看到页面中包含了很多句不同时间格式的文字。这些文字并没有什么特别之处,只是每一句都包含着时间信息。时间信息的格式比较复杂,但是这不影响我们的演示效果。

接下来,我们将介绍实现代码。time标签用于定义公历日期或时间、二十四小时制,时间和时区偏移是可选的。在所有浏览器中,time标签不会渲染任何特殊的效果。但是,它可以让搜索引擎更容易地在网页中找到对应的时间信息。

使用time标签的另一个原因是,世界上有许多不同的日期格式,但是这些不同的格式不容易被电脑识别。如果我们想自动抓取页面上所有事件的日期并将它们插入到日历中,time元素可以让我们附上清晰的可被机器识别的时间或日期。因此,time标签并不是为了给用户看的,而是为了方便搜索引擎更好地在网页上找到对应的时间。

在我们的示例中,时间和普通文字看上去没有任何区别。除了搜索引擎,网页同手机上的日历、提醒等应用程序交互时,time标签也可以提供很大的方便。

time标签非常简单,只包含一个属性datatime,用于规定日期和时间。如果需要,我们还可以通过元素的内容来指定日期和时间。time标签的值有很多种,只要是符合规范的时间写法格式,都可以被接受并转化为第三方使用的格式。

总之,time标签的使用频率并不高,不需要我们进行太多的学习和理解。如果你们知道有这个东西并且知道它的大概意思,就可以了。

今天的分享就到这里,所有的案例和相关文档都可以向我索取。

下期见,想学习编程的同学请关注我。

为开发人员,我们依赖于静态分析工具来检查、lint(分析)和转换我们的代码。我们使用这些工具来帮助我们提高生产效率并生成更好的代码。然而,当我们使用markdown编写内容时,可用的工具就很少。

在本文中,我们将介绍如何开发一个Markdown扩展来解决在使用Markdown管理Django站点中的内容时遇到的挑战。

你认为他们有linter吗?

照片来自Pexels,由mali maeder拍摄

问题

像每个网站一样,我们在主页、FAQ部分和“关于”页面等地方都有不同类型的(大部分)静态内容。很长一段时间以来,我们都是在Django模板中直接管理这些内容的。

当我们最终决定是时候将这些内容从模板转移到数据库中时,我们认为最好使用Markdown。从Markdown生成HTML更安全,它提供了一定程度的控制和一致性,并且对于非技术用户来说更容易处理。随着我们转移过程的进展,我们注意到我们遗漏了一些东西:

内部链接

当URL更改时,链接到内部页面的链接可能会中断。在Django模板和视图中,我们使用了reverseand {% url %},但是这在普通的Markdown中是不可用的。

在不同环境之间进行复制

绝对内部连接不能在不同环境之间进行复制。这可以使用相对链接来解决,不过目前没有开箱即用的增强这一点的方法。

无效链接

无效链接会损害用户体验,并导致用户质疑整个内容的可靠性。这并不是Markdown独有的东西,只不过HTML模板是由对URL有一定了解的开发人员维护的。另一方面,Markdown文档是为非技术写作人员设计的。

前期工作

当我研究这个问题时,我搜索了Python linters、Markdown预处理器和扩展来帮助生成更好的Markdown。结果都不是很好。一个引人注目的方法是使用Django模板来生成Markdown文档。

使用Django模板预处理Markdown

使用Django模板,你可以使用诸如url之类的模板标记来反向查询URL名称,并配合使用条件、变量、日期格式和所有其他Django模板特性。这种方法本质上是使用Django模板作为Markdown文档的预处理程序。

我个人认为这可能不是非技术作家的最佳解决方案。另外,我担心提供对Django模板标记的访问可能是危险的。

使用 Markdown

对这个问题有了更好的理解之后,我们准备在Python中更深入地研究Markdown。

将Markdown转换为HTML

要在Python中开始使用Markdown,我们先安装markdown包:

接着,创建一个Markdown对象并使用其函数将一些Markdown转换成HTML:

你现在可以在你的模板中使用这个HTML代码片段。

使用Markdown扩展

基本的Markdown处理器提供了生成HTML内容的基本要素。对于更“新奇”的选项,Python markdown包包含了一些内置扩展。一个流行的扩展是“extra”扩展,除了其他东西之外,它增加了对隔离代码块的支持:

为了使用我们独特的Django功能扩展Markdown,我们将开发自己的扩展。

创建一个Markdown扩展来处理内联链接

如果你查看源代码,你将看到要将markdown转换为HTML, Markdown会使用多种不同的处理器。一种类型的处理器是内联处理器。内联处理器会匹配特定的内联模式,如链接、反引号、粗体文本和带下划线的文本,并将它们转换为HTML。

我们的Markdown扩展的主要目的是验证和转换链接。因此,我们最感兴趣的内联处理器是LinkInlineProcessor。这个处理器以[Haki的网站](https://hakibenito.com)的形式获取markdown ,解析它并返回一个包含链接和文本的元组。

为了扩展该功能,我们扩展了LinkInlineProcessor并创建了一个Markdown.Extension, 我们用它来处理链接:

我们来将这段代码分解一下::

  • DjangoUrlExtension扩展注册了一个名为DjangoLinkInlineProcessor的内联链接处理器。这个处理器将取代任何其他现有的链接处理器。

  • 内联处理器DjangoLinkInlineProcessor扩展了内置的LinkInlineProcessor,并在它处理的每个链接上调用clean_link函数。

  • clean_link函数接收一个链接和一个域名,并返回一个转换后的链接。这就是我们要插入我们的实现的地方。

如何获得网站域名


要识别到你自己网站的链接,你必须知道你的网站的域名。如果你正在使用Django的sites框架,那么你可以使用它来获取当前域名。


我没有把它包含在我的实现中,因为我们没有使用sites框架。相反,我们在Django设置中设置了一个变量。


获取当前域名的另一种方法是使用HttpRequest对象。如果内容只在你自己的站点中被编辑,你可以尝试从请求对象中插入站点域名。这可能需要对你的实现进行一些更改。

要使用该扩展,请在初始化一个新的Markdown实例时添加它:

太好了,这个扩展已经被使用了,我们准备进入有趣的部分了!

验证和转换Django链接

既然我们得到了在所有链接上调用clean_link的扩展,那我们可以来实现我们的验证和转换逻辑。

验证mailto链接

要开始工作,我们将从一个简单的验证开始。mailto链接对于使用预定义的收件人地址、主题甚至消息正文打开用户的电子邮件客户端非常有用。

一个常见的mailto链接是这样的:

这个链接将打开你的电子邮件客户端,并设置成撰写一封主题行为“我需要帮助!”的新电子邮件给“support@service.com”。

mailto链接不一定非要包含电子邮件地址。如果你看一看这篇文章底部的“分享”按钮,你会发现像这样的一个mailto链接:

这个mailto链接没有包含收件人,仅包含了主题行和消息正文。

既然我们已经很好地理解了mailto链接是什么样子的,我们就可以向clean_link函数添加第一个验证:

为了验证mailto链接,我们向clean_link中添加了以下代码:

  • 检查链接是否以mailto:开头,以识别相关链接。

  • 使用正则表达式将链接分割到它的组件。

  • 从mailto链接中删除实际的电子邮件地址,并使用Django的EmailValidator验证它。

注意,我们还添加了一种名为InvalidMarkdown的新异常类型。我们定义了自己的自定义异常类型,以将它与markdown本身所引发的其他错误区分开来。

自定义错误类

我曾经写过关于自定义错误类的文章,为什么它们是有用的,以及你什么时候应该使用它们。

在我们继续之前,让我们添加一些测试,看看它的实际效果:

太棒了!按预期的运行了。

处理内部和外部链接

既然我们已经了解了mailto链接,我们也可以处理其他类型的链接:

外部链接

  • 我们的Django应用程序外部的链接。

  • 必须包含一个页面跳转协议(scheme):http或https。

  • 理想情况下,我们还希望确保这些链接没有被破坏,但我们现在不会这样做。

内部链接

  • 到我们的Django应用程序中的页面的链接。

  • 链接必须是相对的:这将允许我们在不同环境之间移动内容。

  • 使用Django的URL名称而不是一个URL路径:这将允许我们安全地来回移动视图,而不必担心markdown内容中的失效链接。

  • 链接可能包含查询参数(?)和片段(#)。

SEO

从SEO的角度来看,公共URL不应该改变。当他们这样做的时候,你应该使用重定向正确地处理它,否则你可能会受到搜索引擎的惩罚。

有了这个需求列表,我们就可以开始工作了。

解析URL名称

要链接到内部页面,我们希望编写者提供一个URL名称,而不是URL路径。例如,假设我们有这个视图:

这个页面的URL路径是https://example.com/, URL名称是home。我们想要在我们的markdown链接中使用这个URL名称home,就像这样:

这将渲染到:

我们还想支持查询参数和散列:

这将渲染到以下HTML:

在使用URL名称时,如果我们更改了URL路径,内容中的链接将不会被破坏。要检查作者提供的href是否是一个有效的url_name,我们可以尝试reverse它:

URL名称“home”指向URL路径“/”。当没有匹配项时,将会引发一个异常:

在我们继续之前,当URL名称包含查询参数或散列时,会发生什么:

这是有意义的,因为查询参数和散列不是URL名称的一部分。

要使用reverse并支持查询参数和散列,我们首先需要清除值。然后,检查它是一个有效的URL名称,并返回包含查询参数和散列的URL路径,如果提供了的话:

这个代码段使用一个正则表达式来以?或#的出现对href进行分割,并返回各部分。

请确保它可以工作:

太了不起了!作者们现在可以在Markdown中使用URL名称了。它们还可以包括要添加到该URL的查询参数和片段。

处理外部链接

要正确处理外部链接,我们需要检查两件事:

1.外部链接总是提供一个跳转协议,http:或者https:。

2.阻止到我们自己网站的绝对链接。内部链接应该使用URL名称。

到目前为止,我们已经处理了URL名称和mailto链接。如果我们通过了这两个检查,这意味着href是一个URL。让我们从检查链接是否是链接到我们自己的网站开始:

函数urlparse会返回一个命名元组,该元组包含URL的不同部分。如果netloc属性等于site_domain,那么该链接就确实是一个内部链接。

如果URL实际上是内部的,我们就需要终止。但是,请记住,作者们不一定是技术人员,因此我们希望帮助他们,并提供一个有用的错误消息。我们要求该内部链接使用URL名称而不是URL路径,所以最好让作者们知道他们提供的路径的URL名称。

要获得一个URL路径的URL名称,Django为我们提供了一个名为resolve的函数:

当找到匹配项时,resolve会返回一个ResolverMatch对象,其中包含URL名称和其他信息。当没有找到匹配项时,它就会引发一个错误:

这实际上就是Django在底层所做的工作,用来确定在一个新请求到来时执行哪个视图函数。

为了给作者们提供更好的错误信息,我们可以使用来自ResolverMatch对象的URL名称:

当我们识别出内部链接时,我们要处理两种情况:

  • 我们没有识别出这个URL:这个URL很可能是不正确的。请作者检查该URL是否有错误。

  • 我们识别出了这个URL: 这个URL是正确的,所以就告诉作者应该使用什么URL名称。

我们来实际地看一下它:

漂亮!外部链接被接受,内部链接被拒绝,并带有一个有用的消息。

要求跳转协议

我们要做的最后一件事是确保外部链接包含一个跳转协议,要么是http:,要么是https:。让我们将这最后一部分添加到函数clean_link:

使用解析后的URL,我们可以很容易地检查跳转协议。让我们确保它正在工作:

我们向这个函数提供了一个没有跳转协议的链接,但是它运行失败了,并显示了一条有用的消息。太酷了!

整合代码

这是clean_link函数的全部代码:

要了解所有这些特性的一个实际用例是什么样子的,请看下面的内容:

这将产生以下HTML:

不错!

结论

我们现在有一个很不错的扩展,它可以验证和转换Markdown文档中的链接!现在,在不同环境之间移动文档和保持内容整洁要容易多了,最重要的是,可以保持正确和最新!

源码

你可以在这个gist中找到全部源代码。(地址:https://gist.github.com/hakib/73fccc340e855bb65f42197e298c0c7d )

题外话

本文中所描述的功能对我们很有用,但是你可能需要根据自己的需求对它进行调整。

如果你需要一些想法,那么除了这个扩展之外,我们还创建了一个markdown Preprocessor,它允许作者们在markdown中使用常量。例如,我们定义了一个名为SUPPORT_EMAIL的常量,我们像这样使用它:

该预处理程序将用我们定义的文本替换字符串$SUPPORT_EMAIL,然后才渲染Markdown。

英文原文:https://hakibenita.com/django-markdown
译者:Nothing

者:Peter
来源:Python编程时光

在生活和工作中,我们每个人每天都在和时间打交道:

  • 早上什么时候起床?
  • 地铁几分钟来一趟?
  • 中午什么时候开始午休?
  • 明天是星期几?
  • 距离上次买衣服已经2个月呢?
  • 领导让我给代码加上一个定时任务的功能,怎么办?
  • 不同的情况会遇到不同的时间问题:具体时间点、时间间隔、星期等,无时不刻我们在和时间碰撞。本文将利用Python对时间相关的类,及其方法与属性等进行详细的讲解

    1. 时间戳

    1.1时间戳简介

    在正式讲解时间的相关函数之前,我们必须先一个概念:时间戳。本文中特指unix时间戳。

    时间戳Timestamp是指在一连串的数据中加入辨识文字,如时间或者日期等,用以保障本地数据更新顺序和远程的一致。

    unix时间戳是从1970年1月1日(UTC/GMT的午夜)开始所经过的秒数,不考虑闰秒。1970-01-01就是经常我们在MySQL中时间为空的时候,转化空的时间戳之后得到的时间。一个小时表示为UNIX时间戳格式为:3600秒;一天表示为UNIX时间戳为86400秒,闰秒不计算。具体的对照表如下:

    1.2时间戳转化网站

    下面介绍几个时间戳和具体时间之间相互转化的网站:

    1、站长工具:https://tool.chinaz.com/tools/unixtime.aspx

    2、在线工具:https://tool.lu/timestamp/

    3、Json在线解析:https://www.sojson.com/unixtime.html

    4、Unix时间戳在线转换(菜鸟工具):https://c.runoob.com/front-end/852

    5、北京时间(时间与时间戳互换工具):http://www.beijing-time.org/shijianchuo/

    介绍完时间戳的基本知识,下面重点讲解3个与时间和日期相关的Python库:

  • calendar
  • time
  • datetime
  • 2.calendar

    calendar的中文意思是"日历",所以它其实适合进行日期,尤其是以日历的形式展示。

    2.1模块内容

    下面举例说明:

    2.2calendar

    我们显示即将过去2020年的日历,使用默认的参数:

    import calendar
    year = calendar.calendar(2020)
    print(year)
    


    改变参数再来显示一次:

    year = calendar.calendar(2020,w=3,l=1,c=8)
    print(year)
    


    我们发现整个日历变宽了,而且星期的英文也是3个字母来显示的,解释一下3个参数的含义:

  • c:每月间隔距离
  • w:每日宽度间隔
  • l:每星期行数
  • 其中每行长度为:21*w+18+2*c,3个月一行

    最后,看看即将到来的2021年日历:

    2.3isleap(year)

    该函数的作用是判断某个年份到底是不是闰年。如果是则返回True,否则返回的是False。

    普通年份能够被4整除,但是不能被100整除,称之为普通闰年

    年份是整百数的,必须能够被400整除,称之为世纪闰年

    2.4leapdays(y1,y2)

    判断两个年份之间有多少个闰年,包含y1,但是不包含y2,类似Python切片中的包含头部不包含尾部

    2.5month(year,month,w=2,l=1)

    该函数返回的是year年的month月的日历,只有两行标题,一周一行。每日间隔宽度为w个字符,每行的长度为7*w + 6,其中l是每星期的行数

    首先看看默认效果;

    接下来我们改变w和l两个参数:

    1、改变w,我们发现星期的表示变成了3个字母;同时每天之间的间隔变宽了(左右间隔)

    2、改变参数l,我们发现每个星期之前的间隔(上下)变宽了

    2.6monthcalendar(year,month)

    通过列表的形式返回year年month月的日历,列表中还是列表形式。每个子列表是一个星期。如果没有本月的日期则用0表示。每个子列表都是从星期1开始的,特点概括如下:

  • 每个子列表代表的是一个星期
  • 从星期一到星期日,没有出现在本月的日期用0代替
  • 我们还是以2020年12月份为例:

    和上面的日历进行对比,我们发现:出现0的位置的确是没有出现在12月份中

    我们再看看2020年3月份的日历:

    2.7monthrange(year,month)

    该函数返回的结果是一个元组,元组中有两个数值(a,b)

  • 数值a代表的是该月从星期几开始;规定6代表星期天,取值为0-6
  • 数值b代表该月总共有多少天
  • 通过一个例子来讲解,还是以2020年12月份为例:

    结果中的1表示12月份从星期2开始(0-6,6代表星期日),该月总共31天

    2.8weekday(y,m,d)

    weekday方法是输入年月日,我们便可知道这天是星期几;返回值是0-6,0代表星期1,6代表星期天

    通过一个例子来讲解,以12月12号为例:

    双12是星期六,返回的结果是5,5代表的就是星期六,刚好吻合。

    3.time

    time模块是涉及到时间功能中最常用的一个模块,在Python的相关时间需求中经常会用到,下面具体讲解该模块的使用方法。

    3.1模块内容

    先看模块的整体使用

    3.2time

    time.time()是获取当前的时间,更加严格地说,是获取当前时间的时间戳

    再次理解时间戳:它是以1970年1月1日0时0份0秒为计时起点,计算到当前的时间长度(不考虑闰秒)

    3.3localtime

    time.localtime打印当前的时间,得到的结果是时间元组,具体含义:

    笔记:结果是时间元组

    time.localtime的参数默认是time.time()的时间戳,可以自己输入某个时间戳来获取其对应的时间

  • 默认当前时间戳
  • 指定某个时间戳
  • 3.4gmtime

    localtime()得到的是本地时间,如果需要国际化,使用gmtime(),最好是使用格林威治时间。

    格林威治标准时间:位于英国伦敦郊区的皇家格林威治天文台的标准时间,本初子午线经过那里。

    3.5asctime

    time.asctime的参数为空时,默认是以time.localtime的值为参数,得到当前的日期、时间、星期;另外,我们也可以自己设置参数,参数是时间元组

  • 使用当前时间的默认时间元组localtime
  • 自己指定一个时间元组
  • 获取当前时间的具体时间和日期:

    3.6ctime

    ctime的参数默认是时间戳;如果没有,也可以指定一个时间戳

    3.7mktime

    mktime()也是以时间元组为参数的,它返回的是时间戳,相当于是localtime的逆向过程

    3.8strftime

    strftime()是按照我们指定的格式将时间元组转化为字符串;如果不指定时间元组,默认是当前时间localtime()。常用到的时间格式见下表:

    我们举例说明:

  • 字符串中的分隔符我们可以任意指定
  • 可以同时显示年月日时分秒等
  • 3.9strptime

    strptime()是将字符串转化为时间元组,我们需要特别注意的是,它有两个参数:

  • 待转化的字符串
  • 时间字符串对应的格式,格式就是上面表中提到的
  • 4.datetime

    虽然time模块已经能够解决很多的问题,但是实际工作和业务需求中需要更多的工具,让我们使用起来更方便和快捷,datetime便是其中一个很好用的模块。datetime模块中几个常用的类如下:

  • date:日期类,常用属性:year/month/day
  • time:时间类,常用属性:hour/minute/second/microsecond
  • datetime:日期时间类
  • timedelta:时间间隔,即两个时间点之间的时间长度
  • tzinfo:时区类
  • 4.1模块内容



    4.2date

    首先我们引入date类,并创建一个日期对象:

    1、然后我们可以操作这个日期对象的各种属性:后面加上()

    print("当前日期:",today)  # 当前日期
    print("当前日期(字符串):",today.ctime())   # 返回日期的字符串
    print("时间元组信息:",today.timetuple())   # 当前日期的时间元组信息
    print("年:",today.year)   # 返回today对象的年份
    print("月:",today.month)  # 返回today对象的月份
    print("日:",today.day)   # 返回today对象的日
    print("星期:",today.weekday())  # 0代表星期一,类推
    print("公历序数:",today.toordinal())  # 返回公历日期的序数
    print("年/周数/星期:",today.isocalendar())   # 返回一个元组:一年中的第几周,星期几
    
    # 结果显示
    当前日期: 2020-12-25
    当前日期(字符串):Fri Dec 25 00:00:00 2020
    时间元组信息:time.struct_time(tm_year=2020, tm_mon=12, tm_mday=25, tm_hour=0, tm_min=0, tm_sec=0, tm_wday=4, tm_yday=360, tm_isdst=-1)
    年: 2020
    月: 12
    日: 25
    星期: 4
    公历序数: 737784
    年/周数/星期: (2020, 52, 5)
    

    2、date类中时间和时间戳的转换:

    具体时间的时间戳转成日期:

    3、格式化时间相关,格式参照time模块中的strftime方法

    from datetime import datetime, date, time
    today = date.today()
    
    print(today)
    # 2020-12-26  默认连接符号是-
    
    print(today.strftime("%Y/%m/%d"))  # 指定连接符
    # 2020/12/26
    
    print(today.strftime("%Y:%m:%d"))
    # 2020:12:26
    
    print(today.strftime("%Y/%m/%d %H:%M:%S"))  # 转化为具体的时间
    # 2020/12/26 00:00:00
    

    4、修改日期使用replace方法

    4.3time

    time类也是要生成time对象,包含hour、minute、second、microsecond,我们还是通过例子来学习:

    from datetime import time
    
    t = time(10,20,30,40)
    print(t.hour)  # 时分秒
    print(t.minute)
    print(t.second)
    print(t.microsecond)  # 微秒
    
    # 结果
    10
    20
    30
    40
    

    4.4datetime

    datetime类包含date类和time类的全部信息,下面是类方法相关的:

    from  datetime import datetime
    
    print(datetime.today())
    print(datetime.now())
    print(datetime.utcnow())# 返回当前UTC日期和时间的datetime对象
    print(datetime.fromtimestamp(1697302830))  # 时间戳的datetime对象
    print(datetime.fromordinal(699000) )
    print(datetime.combine(date(2020,12,25), time(11,22,54)))  # 拼接日期和时间
    print(datetime.strptime("2020-12-25","%Y-%m-%d"))
    
    # 结果
    2020-12-25 23:24:42.481845
    2020-12-25 23:24:42.482056
    2020-12-25 15:24:42.482140
    2023-10-15 01:00:30
    1914-10-19 00:00:00
    2020-12-25 11:22:54
    2020-12-25 00:00:00
    

    再看看相关对象和属性相关:

    from datetime import datetime 
    
    d = datetime(2020,12,25,11,24,23)
    
    print(d.date())
    print(d.time())
    print(d.timetz())  # 从datetime中拆分出具体时区属性的time
    print(d.replace(year=2021,month=1))  # 替换
    print(d.timetuple())  # 时间元组
    print(d.toordinal())  # 和date.toordinal一样
    print(d.weekday())
    print(d.isoweekday())
    print(d.isocalendar())
    print(d.isoformat())
    print(d.strftime("%Y-%m-%d :%H:%M:%S"))
    
    # 结果
    2020-12-25
    11:24:23
    11:24:23
    2021-01-25 11:24:23
    time.struct_time(tm_year=2020, tm_mon=12, tm_mday=25, tm_hour=11, tm_min=24, tm_sec=23, tm_wday=4, tm_yday=360, tm_isdst=-1)
    737784
    4
    5
    (2020, 52, 5)
    2020-12-25T11:24:23
    2020-12-25 :11:24:23
    

    4.5timedelta

    timedelta对象表示的是一个时间段,即两个日期date或者日期时间datetime之间的差;支持参数:weeks、days、hours、minutes、seconds、milliseconds、microseconds


    4.6tzinfo

    本地时间指的是我们系统本身设定时区的时间,例如中国处于北京时间,常说的东八区UTC+8:00datetime类有一个时区属性tzinfo

    tzinfo是一个关于时区信息的类,是一个抽象的基类,不能直接被实例化来使用。它的默认值是None,无法区分具体是哪个时区,需要我们强制指定一个之后才能使用。

    因为本身系统的时区刚好在中国处于东八区,所以上述代码是能够正常运行的,结果也是OK的。那如果我们想切换到其他时区的时间,该如何操作呢?这个时候我们需要进行时区的切换。

    1、我们先通过utcnow()获取到当前的UTC时间

    utc_now = datetime.utcnow().replace(tzinfo=timezone.utc)  # 指定utc时区
    print(utc_now)
    
    # 结果
    2020-12-26 01:36:33.975427+00:00
    

    2、通过astimezone()将时区指定为我们想转换的时区,比如东八区(北京时间):

    # 通过astimezone切换到东八区
    
    beijing = utc_now.astimezone(timezone(timedelta(hours=8)))
    print(beijing)
    
    # 结果
    2020-12-26 09:36:33.975427+08:00
    

    用同样的方法切到东九区,东京时间:

    # UTC时区切换到东九区:东京时间
    
    tokyo = utc_now.astimezone(timezone(timedelta(hours=9)))
    print(tokyo)
    
    # 结果
    2020-12-26 10:36:33.975427+09:00
    

    还可以直接从北京时间切换到东京时间

    # 北京时间(东八区)直接切换到东京时间(东九区)
    
    tokyo_new = beijing.astimezone(timezone(timedelta(hours=9)))
    print(tokyo_new)
    
    # 结果
    2020-12-26 10:36:33.975427+09:00
    


    5.常用时间转化

    下面介绍几个工作中用到的时间转化小技巧:

    1. 时间戳转日期
    2. 日期转时间戳
    3. 格式化时间
    4. 指定格式获取当前时间

    5.1时间戳转成日期

    时间戳转成具体时间,我们需要两个函数:

  • time.localtime:将时间戳转成时间元组形式
  • time.strftime:将时间元组数据转成我们需要的形式
  • import time
    now_timestamp = time.time()  # 获取当前时间的时间戳
    
    # 时间戳先转成时间元组,strftime在转成指定格式
    now_tuple = time.localtime(now_timestamp)
    time.strftime("%Y/%m/%d %H:%M:%S", now_tuple)
    
    # 结果
    '2020/12/26 11:19:01'
    

    假设我们指定一个具体的时间戳来进行转换:

    import time
    timestamp = 1608852741  # 指定时间戳
    
    a = time.localtime(timestamp)  # 获得时间元组形式数据
    print("时间元组数据:",a)
    time.strftime("%Y/%m/%d %H:%M:%S", a)  # 格式化
    
    # 结果
    时间元组数据:time.struct_time(tm_year=2020, tm_mon=12, tm_mday=25, tm_hour=7, tm_min=32, tm_sec=21, tm_wday=4, tm_yday=360, tm_isdst=0)
    '2020/12/25 07:32:21'
    

    如果我们不想指定具体的格式,只想获取时间戳对应的时间,直接通过time.ctime即可:

    import time
    time.ctime(1608852741)
    
    # 结果
    'Fri Dec 25 07:32:21 2020'
    

    5.2日期时间转成时间戳

    日期时间转成时间戳格式,我们需要使用两个方法:

  • strptime():将时间转换成时间数组
  • mktime():将时间数组转换成时间戳
  • 通过具体的案例来学习一下:

    date = "2020-12-26 11:45:34"
    
    # 1、时间字符串转成时间数组形式
    date_array = time.strptime(date, "%Y-%m-%d %H:%M:%S")
    
    # 2、查看时间数组数据
    print("时间数组:", date_array)
    
    # 3、mktime时间数组转成时间戳
    time.mktime(date_array)
    
    # 结果
    时间数组:time.struct_time(tm_year=2020, tm_mon=12, tm_mday=26, tm_hour=11, tm_min=45, tm_sec=34, tm_wday=5, tm_yday=361, tm_isdst=-1)
    1608954334.0
    


    5.3格式化时间

    工作需求中有时候给定的时间格式未必是我们能够直接使用,所以可能需要进行格式的转换,需要使用两个方法:

  • strptime():将时间转换成时间数组
  • strftime():重新格式化时间
  • 通过案例来进行学习:

    import time
    
    old = "2020-12-12 12:28:45"
    
    # 1、转换成时间数组
    time_array = time.strptime(old, "%Y-%m-%d %H:%M:%S")
    
    # 2、转换成新的时间格式(20201212-20:28:54)
    new = time.strftime("%Y%m%d-%H:%M:%S",time_array)  # 指定显示格式
    
    print("原格式时间:",old)
    print("新格式时间:",new)
    
    # 结果
    原格式时间: 2020-12-12 12:28:45
    新格式时间: 20201212-12:28:45
    


    5.4指定格式获取当前时间

    为了能够获取到指定格式的当前时间,我们分为3个步骤:

  • time.time():获取当前时间
  • time.localtime():转成时间元组
  • time.strftime():重新格式化时间
  • 通过一个案例来学习:

    # 1、时间戳
    old_time = time.time()
    # 2、时间元组
    time_array = time.localtime(old_time)
    # 3、指定格式输出
    new_time = time.strftime("%Y/%m/%d %H:%M:%S", time_array)
    print(new_time)
    
    # 结果
    2020/12/26 11:56:08
    

    6.总结

    本文通过各种案例详细介绍了Python中关于时间输出和转化的3个模块:calendar、time、datetime,最后总结了4个工作中常用的时间转化技巧,希望对大家掌握Python中的时间输出和转化有所帮助,不再被时间困扰。