用PostgreSQL生成日历表(Calendar Table)
bigegpt 2025-02-03 11:28 8 浏览
前言
前天我们通过一段代码展示了如何通过Python的Pandas包生成一个日历表,如果各位感兴趣可以参考:通过Pandas生成日历表;
不得不说在 DB Engine(https://db-engines.com/en/ranking)排行榜中,PostgreSQL最近几年一直都很稳定,并且在各个云平台中都有非常成熟的PaaS产品供各个企业应用,那么我们今天的目标就是演示一下如何通过#postgresql#来生成类似的日历表。
本文使用的PostgreSQL版本为14.1,Windows Server 2016环境下运行。
一些重要的功能
GENERATE_SERIES
在完成解决方案之前,我们先了解PostgreSQL的一个函数:“GENERATE_SERIES”,通过字面意思应该能大致猜出来,这个函数可以生成一个序列的数据,直白讲就是生成单列的一个数据表。
我们先通过psql命令得到对于此函数的描述,如下图所示:
此函数主要有几个特点:
- 可以返回整型(bigint,int),浮点类型(numeric)序列;
- 可以返回带时区的时间戳类型(timestamp)序列;
- 可以返回不带时区的时间戳类型(timestamp)序列;
简单举个例子:
1.1 返回整型序列
- 当指定起点,终点参数
SELECT GENERATE_SERIES(1, 10);
结果如下所示:
- 当指定起点,终点,步长参数
步长可以理解为跳跃值,从1开始,如果步长为2,那么下一个数字应该是1 + 2 = 3,以此类推;
SELECT GENERATE_SERIES(1, 10, 2);
1.2 返回浮点类型序列
此功能类似于返回整型序列,不同点是传入的可以是带小数位的浮点类型数据,如下图所示:
SELECT GENERATE_SERIES(1.1, 10.9, 0.5);
1.3 返回时间戳(timestamp)序列
此功能可以通过指定起点,终点和步长三个参数,返回一段时间戳的序列数据,如下图所示:
SELECT GENERATE_SERIES('2022-01-01'::TIMESTAMP, '2022-01-31'::TIMESTAMP, '1 DAY') AS datum;
需要注意的是:
- 必须同时指定三个参数,起点,终点,步长;
- 起点和终点参数,必须是时间戳类型(timestamp),如果传入的是日期类型,需要显示转换;
- 步长可以是小时,分钟,秒,天,星期,年等;
日期类型数据操作
我们需要记住这一个操作:日期 + 整数 = 日期,如下面例子所示:
date + integer → date
Add a number of days to a date
date '2022-01-09' + 5 → 2022-01-14
下面我们结合GENERATE_SERIES 函数实现如何得到一个日期类型的序列;
起点:2022-01-01, 终点:2022-01-31;这两个时间点中间间隔了30天,通过代码实现如下:
SELECT '2022-01-01'::DATE + s.a AS datum
FROM GENERATE_SERIES(0, 30) AS s(a);
结果如下图所示:
这种方案的优势是:
- 返回日期类型序列,原函数仅支持传入时间戳类型数据;
- 不需要按日期时间的Interval指定时间间隔,将时间间隔默认为1天。
自定义函数
通过上面的练习,我们已经能够通过传入开始日期和间隔天数得到我们想要的结果。可是如果业务上经常变换开始日期和时间间隔,我们还需要不断的重写SQL语句。为了避免重新改写SQL语句,我们将定义一个函数“get_calendar”,并将“开始日期”(start_dt)和“时间间隔”(days)做为参数传入,从而使我们的结果和语句更加灵活。
代码如下所示:
CREATE OR REPLACE FUNCTION public.get_calendar(
start_dt date,
days integer)
RETURNS TABLE(datum date)
LANGUAGE 'sql'
AS $BODY$
SELECT start_dt + s.a AS datum
FROM GENERATE_SERIES(0, days) AS s(a)
GROUP BY s.a
ORDER BY 1;
$BODY$;
简单测试一下,依然将‘2022-01-01’作为开始日期,时间间隔设置为30天:
SELECT * FROM get_calendar('2022-01-01', 30);
完整代码实现
最终,我们将通过PostgreSQL的大量日期和字符串转换函数,扩展我们的自定义函数“get_calendar”,得到一个完整的日历表,具体代码如下所示。
/* Author: Derek Zhu
Date: 2022-01-08
Purpose: Calendar table practice in PostgreSQL 14.1
Description:
Start date: 2022-01-01
Set days length in 2nd argument of 'Genarate_series' function */
-- FUNCTION: public.get_calendar(date, integer)
-- DROP FUNCTION IF EXISTS public.get_calendar(date, integer);
CREATE OR REPLACE FUNCTION public.get_calendar(
start_dt date,
days integer)
RETURNS TABLE(datum date, year numeric, month numeric, day_of_month numeric, week_of_year numeric, iso_day_of_week numeric, year_calendar_week text, day_of_year numeric, quarter_of_year numeric, quartal text, year_quartal text, day_name text, month_name text, year_month text, year_half integer, leap_year boolean, weekend text, cw_start date, cw_end date, month_start date, month_end date)
LANGUAGE 'sql'
COST 100
VOLATILE PARALLEL UNSAFE
ROWS 1000
AS $BODY$
SELECT
datum,
EXTRACT(YEAR FROM datum) AS "year",
EXTRACT(MONTH FROM datum) AS "month",
EXTRACT(DAY FROM datum) AS day_of_month,
EXTRACT(WEEK FROM datum) AS week_of_year,
-- ISO 8601 day of the week numbering, The day of the week as Monday (1) to Sunday (7)
EXTRACT(ISODOW FROM datum) AS iso_day_of_week,
-- Standard Gregorian day of the week numbering, The day of the week as Sunday (0) to Saturday (6)
-- EXTRACT(DOW FROM datum) AS day_of_week,
-- ISO calendar year and week
TO_CHAR(datum, 'iyyy/IW') AS year_calendar_week,
EXTRACT(DOY FROM datum) AS day_of_year,
EXTRACT(QUARTER FROM datum) AS quarter_of_year,
'Q' || TO_CHAR(datum, 'Q') AS quartal,
TO_CHAR(datum, 'yyyy/"Q"Q') AS year_quartal,
TO_CHAR(datum, 'TMDay') AS day_name,
TO_CHAR(datum, 'TMMonth') AS month_name,
TO_CHAR(datum, 'yyyy/mm') AS year_month,
-- Half year
CASE WHEN EXTRACT(MONTH FROM datum) < 7 THEN 1 ELSE 2 END AS year_half,
-- Leap year
CASE WHEN EXTRACT(YEAR FROM datum) % 4 = 0 THEN TRUE ELSE FALSE END AS leap_year,
-- Weekend
CASE WHEN EXTRACT(ISODOW FROM datum) in (6, 7) THEN 'Weekend' ELSE 'Weekday' END AS weekend,
-- ISO start and end of the week of this date
datum + (1 - EXTRACT(ISODOW FROM datum))::integer AS cw_start,
datum + (7 - EXTRACT(ISODOW FROM datum))::integer AS cw_end,
-- Start and end of the month of this date
datum + (1 - EXTRACT(DAY FROM datum))::integer AS month_start,
((datum + (1 - EXTRACT(DAY FROM datum))::integer + '1 month'::interval)::date - '1 day'::interval)::DATE AS month_end
FROM (
SELECT start_dt + s.a AS datum
FROM GENERATE_SERIES(0, days) AS s(a)
GROUP BY s.a
) AS calendar
ORDER BY 1;
$BODY$;
ALTER FUNCTION public.get_calendar(date, integer)
OWNER TO postgres;
得到2022年全年日历,如下所示:
SELECT * FROM get_calendar('2022-01-01', 364);
通过文本编辑器观察结果,如下所示:
通过Excel观察结果,如下所示:
至此,我们已经完成了所有功能;
总结
通过PostgreSQL生成日历表主要有下面几个注意点:
- ISO8601标准中,一个星期的天数为:Monday (1) ~ Sunday (7);
- 理解并灵活应用GENERATE_SERIES函数生成日期序列;
- 养成模块化思维习惯,将通用的数据操作抽象为函数或方法,能够扩展应用范围;
与前日通过Python Pandas的案例一样,我们最终也将解决方案抽象为一个函数,供后期灵活调用,虽然传入的参数和最终的结果不完全一致,但是整体思路是类似的。
想对自己说的话
PostgreSQL 目前在很多企业都在大量使用,通过PG集群搭建数据仓库平台也是很多企业近些年在努力做的实现,去IOE早已执行多年,使用开源软件替换商用软件也是大势所趋,PG应该被重视起来,对于PG的一些常见和重要的操作,也要应该熟记于心。
至于MySQL和PG选哪个这种神仙打架的事情,真没时间想那么多,纯开源,还是PG吧~
参考资料
https://www.postgresql.org/docs/current/
后续安排
通过Power BI Desktop实现日历表,敬请期待。
相关推荐
- 或者这些Joplin插件也可以帮助你的笔记应用再一次强大
-
写在前面距离上次分享《搭建私有全平台多端同步笔记,群晖NAS自建JoplinServer服务》已过去一段时间,大家是否开始使用起来了呢?如果你和我一样已经使用过Joplin有一段时间了,那或许你也会...
- Three.JS教程4 threejs中的辅助类
-
一、辅助类简介Three.js提供了一些辅助类(Helpers)以帮助我们更容易地调试、可视化场景中的元素。ArrowHelepr:创建箭头辅助器;AxisHelper:创建坐标轴辅助器;BoxH...
- 第2章 还记得点、线、面吗(二)(第二章还能敲钟吗)
-
glbgltf模型(webvrmodel)-gltf模型下载定制,glb模型下载定制,三维项目电商网站在线三维展示,usdz格式,vr模型网,网页VR模型下载,三维模型下载,webgl网页模型下载我...
- 如何检查Linux系统硬件信息?从CPU到显卡,一网打尽!
-
你可能会问:“我为什么要关心硬件信息?”答案很简单:硬件是Linux系统的根基,了解它可以帮你解决很多实际问题。比如:性能调优:知道CPU核心数和内存大小,才能更好地调整程序运行参数。故障排查:系统卡...
- SpriteJS:图形库造轮子的那些事儿
-
从2017年到2020年,我花了大约4年的时间,从零到一,实现了一个可切换WebGL和Canvas2D渲染的,跨平台支持浏览器、SSR、小程序,基于DOM结构和支持响应式的,高...
- 平时积累的FPGA知识点(6)(fpga经典应用100例)
-
平时在FPGA群聊等积累的FPGA知识点,第六期:1万兆网接口,发三十万包,会出现掉几包的情况,为什么?原因:没做时钟约束,万兆网接口的实现,本质上都是高速serdes,用IP的话,IP会自带约束。...
- 芯片逻辑调度框架设计 都需要那些那些软件工具
-
设计芯片逻辑调度框架通常需要使用以下软件工具:1.逻辑设计工具:例如Vivado、Quartus、SynopsysDesignCompiler等,用于设计和实现逻辑电路。2.仿真工具:例如Mo...
- ZYNQ与DSP之间EMIF16通信(正点原子领航者zynq之fpga开发指南v3)
-
本文主要介绍说明XQ6657Z35-EVM高速数据处理评估板ZYNQ与DSP之间EMIF16通信的功能、使用步骤以及各个例程的运行效果。[基于TIKeyStone架构C6000系列TMS320C6...
- 好课推荐:从零开始大战FPGA(从零开始的冒险4399)
-
从零开始大战FPGA引子:本课程为“从零开始大战FPGA”系列课程的基础篇。课程通俗易懂、逻辑性强、示例丰富,课程中尤其强调在设计过程中对“时序”和“逻辑”的把控,以及硬件描述语言与硬件电路相对应的“...
- 业界第一个真正意义上开源100 Gbps NIC Corundum介绍
-
来源:内容由「网络交换FPGA」编译自「FCCM2020」,谢谢。FCCM2020在5月4日开始线上举行,对外免费。我们有幸聆听了其中一个有关100G开源NIC的介绍,我们对该文章进行了翻译,并对其中...
- 高层次综合:解锁FPGA广阔应用的最后一块拼图
-
我们为什么需要高层次综合高层次综合(High-levelSynthesis)简称HLS,指的是将高层次语言描述的逻辑结构,自动转换成低抽象级语言描述的电路模型的过程。所谓的高层次语言,包括C、C++...
- Xilinx文档编号及其内容索引(部分)
-
Xilinx文档的数量非常多。即使全职从事FPGA相关工作,没有几年时间不可能对器件特性、应用、注意事项等等有较为全面的了解。本文记录了我自使用Xilinx系列FPGA以来或精读、或翻阅、或查询过的文...
- Xilinx Vivado联合Modelsim软件仿真
-
引言:Xilinx公司Vivado开发软件自带仿真工具,可以实现一般性能的FPGA软件仿真测试,其测试执行效率以及性能都不如第三方专用仿真软件Modelsim强。本文我们介绍下如何进行Vivado20...
- 体育动画直播是怎么做出来的?从数据到虚拟赛场的科技魔法!
-
你是否见过这样的比赛直播?没有真实球员,却能看梅西带球突破?足球比赛变成动画版,但数据100%真实?电竞比赛用虚拟形象直播,选手操作实时同步?这就是体育动画直播——一种融合实时数据、游戏引擎和AI的...
- Dialogue between CPC and political parties of neighboring countries held in Beijing
-
BEIJING,May26(Xinhua)--TheCommunistPartyofChina(CPC)inDialoguewithPoliticalPartiesof...
- 一周热门
- 最近发表
- 标签列表
-
- mybatiscollection (79)
- mqtt服务器 (88)
- keyerror (78)
- c#map (65)
- resize函数 (64)
- xftp6 (83)
- bt搜索 (75)
- c#var (76)
- mybatis大于等于 (64)
- xcode-select (66)
- mysql授权 (74)
- 下载测试 (70)
- linuxlink (65)
- pythonwget (67)
- androidinclude (65)
- libcrypto.so (74)
- logstashinput (65)
- hadoop端口 (65)
- vue阻止冒泡 (67)
- jquery跨域 (68)
- php写入文件 (73)
- kafkatools (66)
- mysql导出数据库 (66)
- jquery鼠标移入移出 (71)
- 取小数点后两位的函数 (73)