CategoryResourceRepost/极客时间专栏/MySQL 必知必会/实践篇/09 | 时间函数:时间类数据,MySQL是怎么处理的?.md
louzefeng d3828a7aee mod
2024-07-11 05:50:32 +00:00

344 lines
19 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<audio id="audio" title="09 | 时间函数时间类数据MySQL是怎么处理的" controls="" preload="none"><source id="mp3" src="https://static001.geekbang.org/resource/audio/c3/98/c38dd356368fdd7ecec6b5579f447298.mp3"></audio>
你好我是朱晓峰。今天咱们来聊一聊MySQL的时间函数。
顾名思义,时间函数就是用来处理时间的函数。时间,几乎可以说是各类项目中都会存在的数据,项目需求不同,我们需要的时间函数也不一样,比如:
- 如果我们要统计一天之中不同时间段的销售情况就要获取时间值中的小时值这就会用到函数HOUR()
- 要计算与去年同期相比的增长率这就要计算去年同期的日期时间会用到函数DATE_ADD()
- 要计算今天是周几、有没有优惠活动这就要用到函数DAYOFWEEK()了;
- ……
这么多不同类型的时间函数,该怎么选择呢?这节课,我就结合不同的项目需求,来讲一讲不同的时间函数的使用方法,帮助你轻松地处理各类时间数据。
## 获取日期时间数据中部分信息的函数
我先举个小例子。超市的经营者提出,他们希望通过实际的销售数据,了解到一天当中什么时间段卖得好,什么时间段卖得不好,这样他们就可以根据不同时间的销售情况,合理安排商品陈列和人员促销,以实现收益最大化。
要达到这个目标,我们就需要统计一天中每小时的销售数量和销售金额。
这里涉及3组数据分别是销售单头表demo.transactionhead)、销售单明细表 (demo.transactiondetails)和商品信息表demo.goodsmaster为了便于你理解表的结构和表里的记录都是经过简化的
销售单头表包含了销售单的整体信息,包括流水单号、交易时间、收款机编号、会员编号和收银员编号等。
<img src="https://static001.geekbang.org/resource/image/92/ca/925490737131d75b38fe6af861c645ca.jpeg" alt="">
销售单明细表中保存的是交易明细数据,包括商品编号、销售数量、价格、销售金额等。
<img src="https://static001.geekbang.org/resource/image/fd/0b/fdfbf05544e36251b2784259fb3ca00b.jpeg" alt="">
商品信息表主要包括商品编号、条码、商品名称、规格、单位和售价。
<img src="https://static001.geekbang.org/resource/image/44/01/44f29d06fc689edd79e3fe81a39e2d01.jpeg" alt="">
需要注意的是,销售单明细表通过流水编号与销售单头表关联,其中流水编号是外键。通过流水编号,销售单明细表引用销售单头表里的交易时间、会员编号等信息,同时,通过商品编号与商品信息表关联,引用商品信息表里的商品名称等信息。
首先,我们来分析一下“统计一天中每小时的销售数量和销售金额”的这个需求。
要统计一天中每小时的销售情况实际上就是要把销售数据按照小时进行分组统计。那么解决问题的关键就是把交易时间的小时部分提取出来。这就要用到MySQL的日期时间处理函数EXTRACT和HOUR了。
为了获取小时的值我们要用到EXTRACT()函数。**EXTRACTtype FROM date表示从日期时间数据“date”中抽取“type”指定的部分**。
有了这个函数,我们就可以获取到交易时间的小时部分,从而完成一天中每小时的销售数量和销售金额的查询:
```
mysql&gt; SELECT
-&gt; EXTRACT(HOUR FROM b.transdate) AS 时段,
-&gt; SUM(a.quantity) AS 数量,
-&gt; SUM(a.salesvalue) AS 金额
-&gt; FROM
-&gt; demo.transactiondetails a
-&gt; JOIN
-&gt; demo.transactionhead b ON (a.transactionid = b.transactionid)
-&gt; GROUP BY EXTRACT(HOUR FROM b.transdate)
-&gt; ORDER BY EXTRACT(HOUR FROM b.transdate);
+------+--------+--------+
| 时段 | 数量 | 金额 |
+------+--------+--------+
| 9 | 16.000 | 500.00 |
| 10 | 11.000 | 139.00 |
| 11 | 10.000 | 30.00 |
| 12 | 40.000 | 200.00 |
| 13 | 5.000 | 445.00 |
| 15 | 6.000 | 30.00 |
| 17 | 1.000 | 3.00 |
| 18 | 2.000 | 178.00 |
| 19 | 2.000 | 6.00 |
+------+--------+--------+
9 rows in set (0.00 sec)
```
查询的过程是这样的:
1. 从交易时间中抽取小时信息EXTRACT(HOUR FROM b.transdate)
1. 按交易的小时信息分组;
1. 按分组统计销售数量和销售金额的和;
1. 按交易的小时信息排序。
这里我是用“HOUR”提取时间类型DATETIME中的小时信息同样道理你可以用“YEAR”获取年度信息用“MONTH”获取月份信息用“DAY”获取日的信息。如果你需要获取其他时间部分的信息可以参考下[时间单位](https://dev.mysql.com/doc/refman/8.0/en/expressions.html#temporal-intervals)。
这个查询我们也可以通过使用日期时间函数HOUR()来达到同样的效果。**HOURtime表示从日期时间“time”中获取小时部分信息**。
需要注意的是EXTRACT()函数中的“HOUR”表示要获取时间的类型而HOUR()是一个函数HOUR(time)可以单独使用表示返回time的小时部分信息。
我们可以通过在代码中把EXTRACT函数改成HOUR函数来实现相同的功能如下所示
```
mysql&gt; SELECT
-&gt; HOUR(b.transdate) AS 时段, -- 改为使用HOUR函数
-&gt; SUM(a.quantity) AS 数量,
-&gt; SUM(a.salesvalue) AS 金额
-&gt; FROM
-&gt; demo.transactiondetails a
-&gt; JOIN
-&gt; demo.transactionhead b ON (a.transactionid = b.transactionid)
-&gt; GROUP BY HOUR(b.transdate) -- 改写为HOUR函数
-&gt; ORDER BY HOUR(b.transdate);-- 改写为HOUR函数
+------+--------+--------+
| 时段 | 数量 | 金额 |
+------+--------+--------+
| 9 | 16.000 | 500.00 |
| 10 | 11.000 | 139.00 |
| 11 | 10.000 | 30.00 |
| 12 | 40.000 | 200.00 |
| 13 | 5.000 | 445.00 |
| 15 | 6.000 | 30.00 |
| 17 | 1.000 | 3.00 |
| 18 | 2.000 | 178.00 |
| 19 | 2.000 | 6.00 |
+------+--------+--------+
9 rows in set (0.00 sec)
```
除了获取小时信息我们往往还会遇到要统计年度信息、月度信息等情况MySQL也提供了支持的函数。
- YEARdate获取date中的年。
- MONTHdate获取date中的月。
- DAYdate获取date中的日。
- HOURdate获取date中的小时。
- MINUTEdate获取date中的分。
- SECONDdate获取date中的秒。
这些函数的使用方法和提取小时信息的方法一样,我就不多说了,你只要知道这些函数的含义就可以了,下面我再讲一讲计算日期时间的函数。
## 计算日期时间的函数
我先来介绍2个常用的MySQL的日期时间计算函数。
- DATE_ADDdate, INTERVAL 表达式 type表示计算从时间点“date”开始向前或者向后一段时间间隔的时间。“表达式”的值为时间间隔数正数表示向后负数表示向前“type”表示时间间隔的单位比如年、月、日等
- LAST_DAYdate表示获取日期时间“date”所在月份的最后一天的日期。
这两个函数怎么用呢接下来我还是借助咱们项目的实际需求来给你讲解下。假设今天是2020年12月10日超市经营者提出他们需要计算这个月单品销售金额的统计以及与去年同期相比的增长率。
这里的关键点是需要获取2019年12月的销售数据。因此计算2019年12月的起始和截止时间点就是查询的关键。这个时候就要用到计算日期时间函数了。
下面我重点讲解一下如何通过2个计算日期时间函数来计算2019年12月的起始时间和截止时间。
我们先来尝试获取2019年12月份的起始时间。
第一步用DATE_ADD函数获取到2020年12月10日上一年的日期2019年12月10日。
```
mysql&gt; SELECT DATE_ADD('2020-12-10', INTERVAL - 1 YEAR);
+-------------------------------------------+
| DATE_ADD('2020-12-10', INTERVAL - 1 YEAR) |
+-------------------------------------------+
| 2019-12-10 |
+-------------------------------------------+
1 row in set (0.00 sec)
```
第二步获取2019年12月10日这个时间节点开始上个月的日期这样做的目的是方便获取月份的起始时间
```
mysql&gt; SELECT DATE_ADD(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR),INTERVAL - 1 MONTH);
+------------------------------------------------------------------------+
| DATE_ADD(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR),INTERVAL - 1 MONTH) |
+------------------------------------------------------------------------+
| 2019-11-10 |
+------------------------------------------------------------------------+
1 row in set (0.00 sec)
```
第三步获取2019年11月10日这个时间点月份的最后一天继续接近我们的目标2019年12月01日。
```
mysql&gt; SELECT LAST_DAY(DATE_ADD(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR),INTERVAL - 1 MONTH));
+----------------------------------------------------------------------------------+
| LAST_DAY(DATE_ADD(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR),INTERVAL - 1 MONTH)) |
+----------------------------------------------------------------------------------+
| 2019-11-30 |
+----------------------------------------------------------------------------------+
1 row in set (0.00 sec)
```
到这里我们获得了2019年11月30日这个日期。你是不是觉得我们已经达到目的了呢要是这样的话你就错了。因为2019年11月30日可能会有销售的。如果用这个日期作为统计销售额的起始日期你就多算了这一天的销售。怎么办呢我们还要进行下一步。
第四步计算2019年11月30日后一天的日期
```
mysql&gt; SELECT DATE_ADD(LAST_DAY(DATE_ADD(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR),INTERVAL - 1 MONTH)),INTERVAL 1 DAY);
+-----------------------------------------------------------------------------------------------------------+
| DATE_ADD(LAST_DAY(DATE_ADD(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR),INTERVAL - 1 MONTH)),INTERVAL 1 DAY) |
+-----------------------------------------------------------------------------------------------------------+
| 2019-12-01 |
+-----------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)
```
你看我们终于获得了正确的起始日期2019年12月01日。
同样,我们可以用下面的方法,获得截止日期:
```
mysql&gt; SELECT DATE_ADD(LAST_DAY(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR)),INTERVAL 1 DAY);
+------------------------------------------------------------------------------+
| DATE_ADD(LAST_DAY(DATE_ADD('2020-12-10', INTERVAL - 1 YEAR)),INTERVAL 1 DAY) |
+------------------------------------------------------------------------------+
| 2020-01-01 |
+------------------------------------------------------------------------------+
1 row in set (0.00 sec)
```
简单小结下我们可以用DATE_ADD()来计算从某个时间点开始过去或者未来一个时间间隔的时间通过LAST_DAY()函数,获得某个时间节点当月的最后一天的日期。借助它们,我们就可以获取从某个时间节点出发的指定月份的起始日期和截止日期。
除了DATE_ADD()ADDDATE()、DATE_SUB()和SUBDATE()也能达到同样的效果。
- ADDDATE()跟DATE_ADD()用法一致;
- DATE_SUB()SUBDATE()与DATE_ADD()用法类似,方向相反,执行日期的减操作。
## 其他日期时间函数
学习了刚刚的时间函数,我们已经可以应对大部分有关时间的场景了。但是这还不够,有的时候,我们还需要其他的日期时间信息,比如:
- 今天是几月几号,星期几;
- 两个时间点之间隔了几天;
- 把时间按照一定的格式显示出来;
- ……
这时就要用到其他日期时间函数了主要包括CURDATE()、DAYOFWEEK()、DATE_FORMAT和DATEDIFF()。
我来借助一个例子,具体解释下这些函数怎么用。
超市经营者为了吸引顾客经常要进行一些促销活动。具体来讲就是以周为单位按照周中不同的日期进行促销比如周一如何打折、周二如何打折、周末如何打折等。那么如何计算当天的价格呢我们来看下单品促销信息demo.discountrule
<img src="https://static001.geekbang.org/resource/image/cd/cf/cdc62927a60978e2128616fab3697fcf.jpeg" alt="">
这个表中的信息表示单品打折的时间和折扣率:
- 编号是1的商品周一、周三和周五打折折扣率分别是9折、75折和88折
- 编号是2的商品周二、周四和周六打折折扣率分别是5折、65折和8折。
- 周日所有商品打5折。
如果我们想要查到具体的价格我们首先要知道当前的日期以及今天是星期几。这就要用到2个MySQL的时间函数CURDATE和DAYOFWEEK
- CURDATE获取当前的日期。日期格式为“YYYY-MM-DD”也就是年月日的格式。
- DAYOFWEEKdate获取日期“date”是周几。1表示周日2表示周一以此类推直到7表示周六。
假设今天是2021年02月06日通过下面的代码我们就可以查到今天商品的全部折后价格了
```
mysql&gt; SELECT
-&gt; CURDATE() AS 日期,
-&gt; CASE DAYOFWEEK(CURDATE()) - 1 WHEN 0 THEN 7 ELSE DAYOFWEEK(CURDATE()) - 1 END AS 周几,
-&gt; a.goodsname AS 商品名称,
-&gt; a.salesprice AS 价格,
-&gt; IFNULL(b.discountrate,1) AS 折扣率,
-&gt; a.salesprice * IFNULL(b.discountrate, 1) AS 折后价格
-&gt; FROM
-&gt; demo.goodsmaster a
-&gt; LEFT JOIN
-&gt; demo.discountrule b ON (a.itemnumber = b.itemnumber
-&gt; AND CASE DAYOFWEEK(CURDATE()) - 1 WHEN 0 THEN 7 ELSE DAYOFWEEK(CURDATE()) - 1 END = b.weekday);
+------------+------+----------+-------+--------+----------+
| 日期 | 周几 | 商品名称 | 价格 | 折扣率 | 折后价格 |
+------------+------+----------+-------+--------+----------+
| 2021-02-06 | 6 | 书 | 89.00 | 1.00 | 89.0000 |
| 2021-02-06 | 6 | 笔 | 5.00 | 0.80 | 4.0000 |
| 2021-02-06 | 6 | 橡皮 | 3.00 | 1.00 | 3.0000 |
+------------+------+----------+-------+--------+----------+
3 rows in set (0.00 sec)
```
这个查询我们用到了CURDATE函数来获取当前日期也用到了DAYOFWEEK函数来获取当前是周几的信息。由于DAYOFWEEK()函数以周日为1开始计周一是2……周六是7而数据表中是从周一为1开始计算为了对齐我用到了条件判断函数CASE我来解释下这个函数。
MySQL中CASE函数的语法如下
```
CASE 表达式 WHEN 值1 THEN 表达式1 [ WHEN 值2 THEN 表达式2] ELSE 表达式m END
```
在我们这个查询中“表达式”有7种可能的值。通过CASE函数我们可以根据DAYOFWEEK()函数返回的值对每个返回值进行处理从而跟促销信息表中的字段weekday对应。
除了获取特定的日期,咱们还经常需要把日期按照一定的格式显示出来,这就要用到日期时间格式化的函数**DATE_FORMAT()它表示将日期时间“date”按照指定格式显示**。
举个小例子张三希望用24小时制来查看时间那么他就可以通过使用DATE_FORMAT()函数,指定格式“%T”来实现
```
mysql&gt; SELECT DATE_FORMAT(&quot;2020-12-01 13:25:50&quot;,&quot;%T&quot;);
+-----------------------------------------+
| DATE_FORMAT(&quot;2020-12-01 13:25:50&quot;,&quot;%T&quot;) |
+-----------------------------------------+
| 13:25:50 |
+-----------------------------------------+
1 row in set (0.00 sec)
```
李四习惯按照上下午的方式来查看时间同样他可以使用DATE_FORMAT()函数,通过指定格式“%r”来实现
```
mysql&gt; SELECT DATE_FORMAT(&quot;2020-12-01 13:25:50&quot;,&quot;%r&quot;);
+-----------------------------------------+
| DATE_FORMAT(&quot;2020-12-01 13:25:50&quot;,&quot;%r&quot;) |
+-----------------------------------------+
| 01:25:50 PM |
+-----------------------------------------+
1 row in set (0.00 sec
```
格式的详细内容非常丰富,我就不一一介绍了,我给你分享一个[链接](https://dev.mysql.com/doc/refman/8.0/en/date-and-time-functions.html#function_date-format),你可以随时查看一下。
另外一个重要的时间函数是DATEDIFFdate1,date2表示日期“date1”与日期“date2”之间差几天。假如你要计算某段时间的每天交易金额的平均值只需要把起始日期和截止日期传给这个函数就可以得到中间隔了几天。再用总计金额除以这个天数就可以算出来了
```
mysql&gt; SELECT DATEDIFF(&quot;2021-02-01&quot;,&quot;2020-12-01&quot;);
+-------------------------------------+
| DATEDIFF(&quot;2021-02-01&quot;,&quot;2020-12-01&quot;) |
+-------------------------------------+
| 62 |
+-------------------------------------+
1 row in set (0.00 sec)
```
## 总结
今天我们学习了MySQL的时间处理函数包括获取日期时间类型数据中部分信息的函数、计算日期时间的函数和获取特定日期的函数我用图片来帮你汇总了下。
<img src="https://static001.geekbang.org/resource/image/6f/49/6f03f8e150e324101b20acf0c61be649.jpg" alt="">
最后我还想多说一句MySQL中获取的时间其实就是MySQL服务器计算机的系统时间。如果你的系统有一定规模需要在多台计算机上运行就要注意时间校准的问题。比如我们的信息系统受门店经营环境和操作人员的素质所限有时会遇到误操作、停电等故障而导致的计算机系统时间失准问题。这对整个信息系统的可靠性影响非常大。
针对这个问题有2种解决办法。
第一种方法是可以利用Windows系统自带的网络同步的方式来校准系统时间。
另一种办法就是门店统一从总部MySQL服务器获取时间。由于总部的服务器的配置和运维状况一般要好于门店所以系统时间出现误差的可能性也较小。如果采用云服务器系统时间的可靠性会更高。
## 思考题
假如用户想查一下今天是星期几(不能用数值,要用英文显示),你可以写一个简单的查询语句吗?
欢迎在留言区写下你的思考和答案,我们一起交流讨论。如果你觉得今天的内容对你有所帮助,欢迎你把它分享给你的朋友或同事,我们下节课见。