当数据类型是带时区和没有时区时,时间戳值在PostgreSQL中存储不同吗?这些差异可以用简单的测试用例来说明吗?
当前回答
时间戳vs时间戳
Postgres中的timestamp字段基本上只是时间戳字段,其中Postgres实际上只是存储“规范化”的UTC时间,即使输入字符串中给出的时间戳具有时区。
如果您的输入字符串是:2018-08-28T12:30:00+05:30,当此时间戳存储在数据库中时,它将存储为2018-08-28T07:00:00。
与简单的时间戳字段相比,这个字段的优点是你对数据库的输入是独立于时区的,当来自不同时区的应用程序插入时间戳时,或者当你将数据库服务器位置移动到不同的时区时,它不会不准确。
引用文件中的内容:
For timestamp with time zone, the internally stored value is always in UTC (Universal Coordinated Time, traditionally known as Greenwich Mean Time, GMT). An input value that has an explicit time zone specified is converted to UTC using the appropriate offset for that time zone. If no time zone is stated in the input string, then it is assumed to be in the time zone indicated by the system’s TimeZone parameter, and is converted to UTC using the offset for the timezone zone. To give a simple analogy, a timestamptz value represents an instant in time, the same instant for anyone viewing it. But a timestamp value just represents a particular orientation of a clock, which will represent different instances of time based on your timezone.
对于几乎所有用例,时间戳几乎总是一个更好的选择。由于timestamptz和timestamp占用相同的8字节数据,这种选择变得更加容易。
来源: https://hasura.io/blog/postgres-date-time-data-types-on-graphql-fd926e86ee87/
其他回答
这些差异在PostgreSQL文档中有关于日期/时间类型的说明。是的,TIME或TIMESTAMP的处理在带TIME ZONE或不带TIME ZONE之间是不同的。它不会影响值的存储方式;它影响了它们的解释方式。
时区对这些数据类型的影响在文档中有详细介绍。差异来自于系统可以合理地知道的值:
将时区作为值的一部分,该值可以在客户端中作为本地时间呈现。 如果没有时区作为值的一部分,显然默认时区是UTC,因此它是按照该时区呈现的。
行为的不同取决于至少三个因素:
客户端的时区设置。 值的数据类型(即带时区或不带时区)。 是否使用特定的时区指定该值。
以下是这些因素组合的例子:
foo=> SET TIMEZONE TO 'Japan';
SET
foo=> SELECT '2011-01-01 00:00:00'::TIMESTAMP;
timestamp
---------------------
2011-01-01 00:00:00
(1 row)
foo=> SELECT '2011-01-01 00:00:00'::TIMESTAMP WITH TIME ZONE;
timestamptz
------------------------
2011-01-01 00:00:00+09
(1 row)
foo=> SELECT '2011-01-01 00:00:00+03'::TIMESTAMP;
timestamp
---------------------
2011-01-01 00:00:00
(1 row)
foo=> SELECT '2011-01-01 00:00:00+03'::TIMESTAMP WITH TIME ZONE;
timestamptz
------------------------
2011-01-01 06:00:00+09
(1 row)
foo=> SET TIMEZONE TO 'Australia/Melbourne';
SET
foo=> SELECT '2011-01-01 00:00:00'::TIMESTAMP;
timestamp
---------------------
2011-01-01 00:00:00
(1 row)
foo=> SELECT '2011-01-01 00:00:00'::TIMESTAMP WITH TIME ZONE;
timestamptz
------------------------
2011-01-01 00:00:00+11
(1 row)
foo=> SELECT '2011-01-01 00:00:00+03'::TIMESTAMP;
timestamp
---------------------
2011-01-01 00:00:00
(1 row)
foo=> SELECT '2011-01-01 00:00:00+03'::TIMESTAMP WITH TIME ZONE;
timestamptz
------------------------
2011-01-01 08:00:00+11
(1 row)
下面是一个应该会有帮助的例子。如果您有一个带有时区的时间戳,您可以将该时间戳转换为任何其他时区。如果你没有一个基本时区,它将不会被正确转换。
SELECT now(),
now()::timestamp,
now() AT TIME ZONE 'CST',
now()::timestamp AT TIME ZONE 'CST'
输出:
-[ RECORD 1 ]---------------------------
now | 2018-09-15 17:01:36.399357+03
now | 2018-09-15 17:01:36.399357
timezone | 2018-09-15 08:01:36.399357
timezone | 2018-09-16 02:01:36.399357+03
我试图比参考的PostgreSQL文档更容易理解地解释它。
TIMESTAMP变量都不存储时区(或偏移量),尽管它们的名称暗示了这一点。区别在于对存储数据的解释(以及预期的应用程序),而不是存储格式本身:
TIMESTAMP WITHOUT TIME ZONE stores local date-time (aka. wall calendar date and wall clock time). Its time zone is unspecified as far as PostgreSQL can tell (though your application may knows what it is). Hence, PostgreSQL does no time zone related conversion on input or output. If the value was entered into the database as '2011-07-01 06:30:30', then no mater in what time zone you display it later, it will still say year 2011, month 07, day 01, 06 hours, 30 minutes, and 30 seconds (in some format). Also, any offset or time zone you specify in the input is ignored by PostgreSQL, so '2011-07-01 06:30:30+00' and '2011-07-01 06:30:30+05' are the same as just '2011-07-01 06:30:30'. For Java developers: it's analogous to java.time.LocalDateTime. TIMESTAMP WITH TIME ZONE stores a point on the UTC time line. How it looks (how many hours, minutes, etc.) depends on your time zone, but it always refers to the same "physical" instant (like the moment of an actual physical event). The input is internally converted to UTC, and that's how it's stored. For that, the offset of the input must be known, so when the input contains no explicit offset or time zone (like '2011-07-01 06:30:30') it's assumed to be in the current time zone of the PostgreSQL session, otherwise the explicitly specified offset or time zone is used (as in '2011-07-01 06:30:30+05'). The output is displayed converted to the current time zone of the PostgreSQL session. For Java developers: It's analogous to java.time.Instant (with lower resolution though), but with JDBC and JPA 2.2 you are supposed to map it to java.time.OffsetDateTime (or to java.util.Date or java.sql.Timestamp of course).
Some say that both TIMESTAMP variations store UTC date-time. Kind of, but it's confusing to put it that way in my opinion. TIMESTAMP WITHOUT TIME ZONE is stored like a TIMESTAMP WITH TIME ZONE, which rendered with UTC time zone happens to give the same year, month, day, hours, minutes, seconds, and microseconds as they are in the local date-time. But it's not meant to represent the point on the time line that the UTC interpretation says, it's just the way the local date-time fields are encoded. (It's some cluster of dots on the time line, as the real time zone is not UTC; we don't know what it is.)
时间戳vs时间戳
Postgres中的timestamp字段基本上只是时间戳字段,其中Postgres实际上只是存储“规范化”的UTC时间,即使输入字符串中给出的时间戳具有时区。
如果您的输入字符串是:2018-08-28T12:30:00+05:30,当此时间戳存储在数据库中时,它将存储为2018-08-28T07:00:00。
与简单的时间戳字段相比,这个字段的优点是你对数据库的输入是独立于时区的,当来自不同时区的应用程序插入时间戳时,或者当你将数据库服务器位置移动到不同的时区时,它不会不准确。
引用文件中的内容:
For timestamp with time zone, the internally stored value is always in UTC (Universal Coordinated Time, traditionally known as Greenwich Mean Time, GMT). An input value that has an explicit time zone specified is converted to UTC using the appropriate offset for that time zone. If no time zone is stated in the input string, then it is assumed to be in the time zone indicated by the system’s TimeZone parameter, and is converted to UTC using the offset for the timezone zone. To give a simple analogy, a timestamptz value represents an instant in time, the same instant for anyone viewing it. But a timestamp value just represents a particular orientation of a clock, which will represent different instances of time based on your timezone.
对于几乎所有用例,时间戳几乎总是一个更好的选择。由于timestamptz和timestamp占用相同的8字节数据,这种选择变得更加容易。
来源: https://hasura.io/blog/postgres-date-time-data-types-on-graphql-fd926e86ee87/
执行以下命令,查看pgAdmin中的diff:
create table public.testts (tz timestamp with time zone, tnz timestamp without time zone);
insert into public.testts values(now(), now());
select * from public.testts;
如果你在Angular / Typescript / Node API / PostgreSql环境中有类似的时间戳精度问题,希望我完整的回答和解决方案能帮助你。
推荐文章
- 熊猫能自动从CSV文件中读取日期吗?
- 如何添加“删除级联”约束?
- 如何在TypeScript中声明一个固定长度的数组
- 如何将PostgreSQL从9.6版本升级到10.1版本而不丢失数据?
- 为现有数据库生成ERD
- PostgreSQL错误:由于与恢复冲突而取消语句
- 在Java中转换float为String和String为float
- 按IN值列表排序
- 如何在Postgres 9.4中对JSONB类型的列执行更新操作
- PostgreSQL列名区分大小写吗?
- postgresql COUNT(DISTINCT…)非常慢
- Postgres:检查数组字段是否包含值?
- 如何在postgresql中创建数据库用户?
- 使用PSQL命令查找主机名和端口
- Postgresql列表和排序表的大小