postgresql timestamp timestamptz 使用注意事项

原创,转载请注明出处!

postgresql日期时间类型分为timestamp [ (p) ] [ without time zone ]和timestamp [ (p) ] with time zone,它们的区别在于一个无时区,另一有时区,存储大小均为8字节.

同时也要注意time [ (p) ] [ without time zone ]和time [ (p) ] with time zone类型,无时区存储大小为8字节,有时区存储大小为12字节.

postgresql中日期时间类型缩写:
timestamp
timestamptz
date (仅年月日部份,存储大小为4字节)
time
timetz
interval(表示间隔,存储大小为16字节)

interval可以通过函数make_interval来创建,支持年\月\星期\天\小时\分钟\秒的间隔.如now() + make_interval(days => 10)表示10天后,now() - make_interval(secs => 30)表示30秒之前

为保证数据的准确性在保存\使用\计算过程中应尽量使用timestamptz和timetz,尽量避免使用timestamp和time

UTC+8(EAT - 东亚标准时间/中国标准时间(BJT))

注意整个中国使用相同的时区,这就使得这个时区特别的大。在中国最西部的地区,太阳最高的时候是下午3点,在最东部是上午11点。单纯从地理规划来看,整个中国横跨了从东五区(UTC+5)到东九区(UTC+9)共计五个时区。

在postgresql中查询时区的定义

select * from pg_timezone_names

或查询东亚/中国时区定义

select * from pg_timezone_names where utc_offset = '+08:00:00';

查询结果中的"PRC","Asia/Shanghai","Asia/Chongqing"均表示中国

生成一个日期时间示例

select  make_timestamptz(1970,1,1,0,0,0.0,'Asia/Shanghai') 

注意避免使用timestamp类型相关函数,如:make_timestamp

[日期/时间类型]https://www.postgresql.org/docs/current/static/datatype-datetime.html
[日期/时间函数和操作符]https://www.postgresql.org/docs/current/static/functions-datetime.html

图片来源,[维基]https://zh.wikipedia.org/wiki/%E6%97%B6%E5%8C%BA%E5%88%97%E8%A1%A8
图片来源,[维基]https://zh.wikipedia.org/wiki/%E6%97%B6%E5%8C%BA%E5%88%97%E8%A1%A8
### 关于 `TIMESTAMP WITH TIME ZONE` 的示例 在 PostgreSQL 中,`TIMESTAMP WITH TIME ZONE` 是一种用于存储带有时区信息的时间戳的数据类型。这种类型的字段会自动将输入的时间转换为 UTC 存储,并在查询时根据客户端的时区设置返回相应的时间。 #### 创建表并插入数据 下面是一个创建包含 `TIMESTAMP WITH TIME ZONE` 类型列的表的例子: ```sql CREATE TABLE event_logs ( id SERIAL PRIMARY KEY, event_time TIMESTAMPTZ ); ``` 当向该表插入数据时,可以显式指定时区,或者依赖系统的默认时区设置[^1]。 #### 插入带有特定时区的数据 假设我们需要记录某个事件发生的具体时间,并且知道它发生在太平洋标准时间(PST),可以通过以下方式插入数据: ```sql INSERT INTO event_logs (event_time) VALUES ('2023-03-15 14:30:00 PST'); ``` 这里的 `'2023-03-15 14:30:00 PST'` 表示事件发生的当地时间以及对应的时区。PostgreSQL 会在内部将其转换为 UTC 时间进行存储[^3]。 #### 查询并转换时区 如果希望以不同的时区展示这些数据,可以使用 `AT TIME ZONE` 函数来进行转换。例如,要将以 UTC 存储的时间转换为协调世界时(UTC): ```sql SELECT event_time AT TIME ZONE 'UTC' AS utc_event_time FROM event_logs; ``` 同样地,也可以将时间转换回原始的 Pacific Standard Time (PST)或其他任何目标时区: ```sql SELECT event_time AT TIME ZONE 'PST' AS pst_event_time FROM event_logs; ``` 通过这种方式,能够灵活处理不同用户的本地化需求。 #### 当前时间和日期函数 为了获取当前时间并确保其包含了正确的时区信息,可利用内置的时间函数如 `CURRENT_TIMESTAMP` 或者组合使用 `NOW()` 和 `TIMEZONE` 来实现这一点: ```sql SELECT CURRENT_TIMESTAMP AS current_timestamp_with_timezone; -- OR SELECT NOW() AT TIME ZONE 'UTC'; ``` 以上命令分别展示了如何获得当前时刻的同时保留完整的时区上下文[^4]。 --- ### 注意事项 对于 MySQL 用户来说,虽然也有类似的 `TIMESTAMP` 字段支持自动调整基于服务器配置文件定义好的区域设定;但是需要注意的是,默认情况下它的行为可能并不完全相同——即不会真正保存额外附加到每条记录上的独立偏移量而是仅仅依据全局参数决定最终呈现形式[^2]。 因此,在跨平台迁移应用逻辑之前务必仔细阅读官方文档确认差异之处以免造成误解或错误操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

kmblack1

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值