1. 首页
  2. mysql基础

08-八、傻傻分不清的 DATE、DATETIME 和 TIMESTAMP ( 下 )

前一章节我们有提到 MySQL 服务的系统变量 explicit_defaults_for_timestamptimestamp 类型的影响,知道禁用该系统变量时,timestamp 默认会添加 NOT NUTLL 的约束,此时,就不再接受 NULL 值了,同时会使用当前时间戳来填充。

这种情况下,如果要允许 timestamp 类型能够接受 NULL 值,则需要创建表结构时显式的指定 NULL 约束。

但一旦这么修改,那么默认值也会被设置为 NULL,除非使用 default 约束显式的指定默认值为其它常量值。

当然了,可以直接使用 NULL DEFAULT NULL 显示的指定 NULL 作为 timestamp 列的默认值

但是,这里有一个坑,就是不能直接使用 DEFAULT NULLtimestamp 列添加 NULL 作为默认值,因为这会和默认的 NOT NULL 约束冲突

是不是很绕口?

也就是在禁用系统变量 explicit_defaults_for_timestamp 的情况下,会发生以下事情

1、 timestamp 列默认会添加 NOT NULL 约束,并使用 CURRENT_TIMESTAMP() 返回的当前时间戳作为默认值
2、 可以显式的使用 NULL 约束将 timestamp 列改成接受 NULL 值,但这样默认值也会被设置为 NULL
3、 可以在显式的使用 NULL 约束的同时可以使用 DEFAULT 约束,这样,timestamp 列既可以接受 NULL 值,有可以设置其它常量值为默认值。

如果 timestamp 列添加了 NULL 约束,那么我们可以在 SQL 语句中使用 NULL 作为 timestamp 列的值,这样,timestamp 列的值就不会自动填充为当前时间戳了。

例如下面的建表语句,我们添加了几个允许 NULL 值的 timestamp


CREATE TABLE t ( ts1 TIMESTAMP NULL DEFAULT NULL, ts2 TIMESTAMP NULL DEFAULT 0, ts3 TIMESTAMP NULL DEFAULT CURRENT_TIMESTAMP );

在这个表 t 中,允许 NULL 值的 TIMESTAMP 列在插入时并不会自动填充当前时间戳,除非触发了以下规则

  • ts3 列一样添加了 DEFAULT CURRENT_TIMESTAMP 默认值约束,这样只要未显式指定该列的值,那么就会自动填充为当前时间戳。
  • 在插入数据或者更新数据时,使用 CURRENT_TIMESTAMP() 或任何其它相同的函数,比如 NOW() 显式的指定其值。

换句话说,定义为允许 NULL 值的 TIMESTAMP 列仅在添加了 DEFAULT CURRENT_TIMESTAMP 约束时才会自动初始化


CREATE TABLE t (ts TIMESTAMP NULL DEFAULT CURRENT_TIMESTAMP);

如果 TIMESTAMP 列允许 NULL 值但其并没有添加 DEFAULT CURRENT_TIMESTAMP 约束,则必须显式插入与当前日期和时间对应的值。

例如下面两个建表语句,表 t1t2 都添加了这些约束


CREATE TABLE t1 (ts TIMESTAMP NULL DEFAULT '0000-00-00 00:00:00'); CREATE TABLE t2 (ts TIMESTAMP NULL DEFAULT NULL);

为了在任一表中为 TIMESTAMP 列在插入数据时填充当前的时间戳,需要显式为其分配该值。例如


INSERT INTO t2 VALUES (CURRENT_TIMESTAMP); INSERT INTO t1 VALUES (NOW());

说了这么多系统变量 explicit_defaults_for_timestamp 禁用的情况,但如果开启了又会怎么样呢?

如果启用了 explicit_defaults_for_timestamp ,那么仅当添加了 NULL 约束时,TIMESTAMP 列才允许 NULL 值。

而且,TIMESTAMP 列不允许分配 NULL 以分配当前时间戳,无论是使用 NULL 约束还是使用 NOT NULL 约束。

要为 TIMESTAMP 列分配当前时间戳,则需要给列添加 CURRENT_TIMESTAMP() 或其它同义词,例如 NOW()

简单直白一点,就是,如果启用了 explicit_defaults_for_timestamp ,那么 TIMESTAMP 列是无论如何都不能填充 NULL 值的,而且一定要显式的使用 CURRENT_TIMESTAMP() 等同义词函数

希望读者能够给小编留言,也可以点击[此处扫下面二维码关注微信公众号](https://www.ycbbs.vip/?p=28 "此处扫下面二维码关注微信公众号")

看完两件小事

如果你觉得这篇文章对你挺有启发,我想请你帮我两个小忙:

  1. 关注我们的 GitHub 博客,让我们成为长期关系
  2. 把这篇文章分享给你的朋友 / 交流群,让更多的人看到,一起进步,一起成长!
  3. 关注公众号 「方志朋」,公众号后台回复「666」 免费领取我精心整理的进阶资源教程
  4. JS中文网,Javascriptc中文网是中国领先的新一代开发者社区和专业的技术媒体,一个帮助开发者成长的社区,是给开发者用的 Hacker News,技术文章由为你筛选出最优质的干货,其中包括:Android、iOS、前端、后端等方面的内容。目前已经覆盖和服务了超过 300 万开发者,你每天都可以在这里找到技术世界的头条内容。

    本文著作权归作者所有,如若转载,请注明出处

    转载请注明:文章转载自「 Java极客技术学习 」https://www.javajike.com

    标题:08-八、傻傻分不清的 DATE、DATETIME 和 TIMESTAMP ( 下 )

    链接:https://www.javajike.com/article/1624.html

« 09-九、臭名昭著的 MySQL ENUM 类型 ( 上 )
07-七、傻傻分不清的 DATE、DATETIME 和 TIMESTAMP ( 中 )»

相关推荐

QR code