如何设置mysql datetime列的默认值?

How do you set a default value for a MySQL Datetime column?

如何设置mysql datetime列的默认值?

在SQL Server中,它是getdate()。MySQL的等价性是什么?如果这是一个因素的话,我使用的是MySQL5.x。


重要编辑:现在可以用MySQL5.6.5以来的日期时间字段来实现这一点,请看下面的另一篇文章…

以前的版本不能用datetime来实现这一点…

但您可以使用时间戳:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
mysql> create table test (str varchar(32), ts TIMESTAMP DEFAULT CURRENT_TIMESTAMP);
Query OK, 0 rows affected (0.00 sec)

mysql> desc test;
+-------+-------------+------+-----+-------------------+-------+
| Field | Type        | Null | Key | Default           | Extra |
+-------+-------------+------+-----+-------------------+-------+
| str   | varchar(32) | YES  |     | NULL              |       |
| ts    | timestamp   | NO   |     | CURRENT_TIMESTAMP |       |
+-------+-------------+------+-----+-------------------+-------+
2 rows in set (0.00 sec)

mysql> insert into test (str) values ("demo");
Query OK, 1 row affected (0.00 sec)

mysql> select * from test;
+------+---------------------+
| str  | ts                  |
+------+---------------------+
| demo | 2008-10-03 22:59:52 |
+------+---------------------+
1 row in set (0.00 sec)

mysql>

**注意:如果您定义了一个当前时间戳为默认值的列,则需要始终为此列指定一个值,否则更新时该值将自动重置为"now()"。这意味着,如果不想更改该值,则UPDATE语句必须包含"[您的列名]=[您的列名]"(或其他一些值),否则该值将变为"now()"。很奇怪,但是真的。我希望这有帮助。我使用的是5.5.56-mariadb**


在版本5.6.5中,可以在datetime列上设置默认值,甚至可以创建一个列,该列将在行更新时更新。类型定义:

1
2
3
4
CREATE TABLE foo (
    `creation_time`     DATETIME DEFAULT CURRENT_TIMESTAMP,
    `modification_time` DATETIME ON UPDATE CURRENT_TIMESTAMP
)

参考文献:http://optimize-this.blogspot.com/2012/04/datetime-default-now-finally-available.html


MySQL(5.6.5版本之前)不允许将函数用于默认的日期时间值。时间戳因其行为古怪而不适用,不建议用作输入数据。(请参见MySQL数据类型默认值。)

也就是说,您可以通过创建触发器来实现这一点。

我有一个表,其中的日期创建字段类型为datetime。我在那个表上创建了一个触发器"before insert"和"SET NEW.DateCreated=NOW()",它工作得很好。

我希望这对某人有帮助。


对我来说,触发方法是最有效的,但我发现了一个障碍。考虑使用基本触发器将日期字段设置为插入时的当前时间:

1
2
CREATE TRIGGER myTable_OnInsert BEFORE INSERT ON `tblMyTable`
    FOR EACH ROW SET NEW.dateAdded = NOW();

这通常很好,但假设您希望通过insert语句手动设置字段,例如:

1
INSERT INTO tblMyTable(name, dateAdded) VALUES('Alice', '2010-01-03 04:30:43');

发生的情况是,触发器会立即覆盖为字段提供的值,因此设置非当前时间的唯一方法是使用后续更新语句--yuck!若要在提供值时重写此行为,请使用ifnull运算符尝试此稍微修改的触发器:

1
2
CREATE TRIGGER myTable_OnInsert BEFORE INSERT ON `tblMyTable`
    FOR EACH ROW SET NEW.dateAdded = IFNULL(NEW.dateAdded, NOW());

这两个方面都是最好的:您可以为日期列提供一个值,它需要,否则它将默认为当前时间。它仍然是相对于表定义中的默认getDate()之类的干净东西的贫民区,但是我们越来越接近了!


我可以使用表中的这个alter语句来解决这个问题,该语句有两个datetime字段。

1
2
3
ALTER TABLE `test_table`
  CHANGE COLUMN `created_dt` `created_dt` TIMESTAMP NOT NULL DEFAULT '0000-00-00 00:00:00',
  CHANGE COLUMN `updated_dt` `updated_dt` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP;

这与您希望now()函数正常工作一样。插入空值或忽略创建的和更新的字段会在两个字段中产生一个完美的时间戳值。行的任何更新都会更改更新后的日期。如果您通过MySQL查询浏览器插入记录,则还需要一个步骤,即用新的时间戳处理创建的日期的触发器。

1
2
CREATE TRIGGER trig_test_table_insert BEFORE INSERT ON `test_table`
    FOR EACH ROW SET NEW.created_dt = NOW();

触发器可以是你想要的任何东西,我只是喜欢命名约定[trig]uuuu table_u name]uuu[insert]


你可以用触发器来做这类事情。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
CREATE TABLE `MyTable` (
`MyTable_ID`  int UNSIGNED NOT NULL AUTO_INCREMENT ,
`MyData`  varchar(10) NOT NULL ,
`CreationDate`  datetime NULL ,
`UpdateDate`  datetime NULL ,
PRIMARY KEY (`MyTable_ID`)
)
;

CREATE TRIGGER `MyTable_INSERT` BEFORE INSERT ON `MyTable`
FOR EACH ROW BEGIN
        -- Set the creation date
    SET new.CreationDate = now();

        -- Set the udpate date
    Set new.UpdateDate = now();
END;

CREATE TRIGGER `MyTable_UPDATE` BEFORE UPDATE ON `MyTable`
FOR EACH ROW BEGIN
        -- Set the udpate date
    Set new.UpdateDate = now();
END;

对于那些在MySQL中试图设置默认日期时间值而失去信心的人来说,我完全知道你的感受。所以这里是:

1
ALTER TABLE  `table_name` CHANGE `column_name` DATETIME NOT NULL DEFAULT 0

仔细观察,我没有在0左右添加单引号/双引号

我真的在解决这个问题之后跳了起来:d


MySQL5.6已经解决了这个问题。

1
ALTER TABLE mytable CHANGE mydate datetime NOT NULL DEFAULT 'CURRENT_TIMESTAMP'

这确实是一个糟糕的消息。这里有一个长期悬而未决的错误/功能请求。该讨论还讨论了时间戳数据类型的局限性。

我真的很想知道实现这一点有什么问题。


如果已经创建了表,则可以使用

将默认值更改为当前日期时间

1
2
ALTER TABLE <TABLE_NAME>
CHANGE COLUMN <COLUMN_NAME> <COLUMN_NAME> DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP;

将默认值更改为"2015-05-11 13:01:01"

1
2
ALTER TABLE <TABLE_NAME>
CHANGE COLUMN <COLUMN_NAME> <COLUMN_NAME> DATETIME NOT NULL DEFAULT '2015-05-11 13:01:01';

定义多行触发器时,必须将分隔符更改为分号,MySQL编译器将使用分号作为触发器的结尾并生成错误。例如

1
2
3
4
5
6
7
DELIMITER //
CREATE TRIGGER `MyTable_UPDATE` BEFORE UPDATE ON `MyTable`
FOR EACH ROW BEGIN
        -- Set the udpate date
    Set new.UpdateDate = now();
END//
DELIMITER ;

对于所有使用timestamp列作为解决方案的用户,我希望在手册中遵循以下限制:

http://dev.mysql.com/doc/refman/5.0/en/datetime.html

"timestamp数据类型的范围是'1970-01-01 00:00:01'UTC到'2038-01-19 03:14:07'UTC。它有不同的属性,这取决于MySQL版本和服务器运行的SQL模式。这些属性将在本节后面介绍。"

所以这显然会在28年内破坏你的软件。

我相信数据库方面唯一的解决方案是使用类似于其他答案中提到的触发器。


可以使用now()设置日期时间列的值,但请记住,不能将其用作默认值。


我运行的是MySQLServer5.7.11,这句话是:

1
ALTER TABLE table_name CHANGE date_column datetime NOT NULL DEFAULT '0000-00-00 00:00:00'

不工作。但以下内容:

1
ALTER TABLE table_name CHANGE date_column datetime NOT NULL DEFAULT '1000-01-01 00:00:00'

只是工作。

作为旁注,MySQL文档中提到:

The DATE type is used for values with a date part but no time part. MySQL retrieves and displays DATE values in 'YYYY-MM-DD' format. The supported range is '1000-01-01' to '9999-12-31'.

即使他们也说:

Invalid DATE, DATETIME, or TIMESTAMP values are converted to the"zero" value of the appropriate type ('0000-00-00' or '0000-00-00 00:00:00').


虽然您不能在默认定义中使用DATETIME,但是您可以在insert语句中简单地合并一个select语句,如下所示:

1
INSERT INTO Yourtable (Field1, YourDateField) VALUES('val1', (select now()))

注意表中没有引号。

对于MySQL 5.5


下面是如何在MySQL5.1上执行此操作:

1
2
ALTER TABLE `table_name` CHANGE `column_name` `column_name`
TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP;

我不知道你为什么要输入两次列名。


如果您试图将默认值设置为now(),我认为MySQL不支持该设置。在MySQL中,除了timestamp数据类型列之外,不能将函数或表达式用作任何类型的列的默认值,您可以将当前的时间戳指定为默认值。


我认为在MySQL中很简单,因为mysql内置函数now()给出了当前时间(插入的时间)。

所以您的查询应该类似

1
2
3
4
CREATE TABLE defaultforTime(
    `creation_time`     DATETIME DEFAULT CURRENT_TIMESTAMP,
    `modification_time` DATETIME default now()
);

谢谢您。


1
2
3
4
CREATE TABLE `testtable` (
    `id` INT(10) NULL DEFAULT NULL,
    `colname` DATETIME NULL DEFAULT '1999-12-12 12:12:12'
)

在上面创建"testtable"的查询中,我使用"1999-12-12 12:12:12"作为日期时间列colname的默认值。


使用以下代码

1
2
3
4
5
6
7
8
DELIMITER $$

    CREATE TRIGGER bu_table1_each BEFORE UPDATE ON table1 FOR EACH ROW
    BEGIN
      SET new.datefield = NOW();
    END $$

    DELIMITER ;

这是我的触发器示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/************ ROLE ************/
drop table if exists `role`;
create table `role` (
    `id_role` bigint(20) unsigned not null auto_increment,
    `date_created` datetime,
    `date_deleted` datetime,
    `name` varchar(35) not null,
    `description` text,
    primary key (`id_role`)
) comment='';

drop trigger if exists `role_date_created`;
create trigger `role_date_created` before insert
    on `role`
    for each row
    set new.`date_created` = now();


例如,如果我有一个名为"site"的表,其中有一个created_at和一个update_at列,该列都是datetime,并且需要now的默认值,那么我可以执行以下SQL来实现这一点。

1
2
3
4
5
6
7
ALTER TABLE `site` CHANGE `created_at` `created_at` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP;

ALTER TABLE `site` CHANGE `created_at` `created_at` DATETIME  NULL DEFAULT NULL;

ALTER TABLE `site` CHANGE `updated_at` `updated_at` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP;

ALTER TABLE `site` CHANGE `updated_at` `updated_at` DATETIME NULL DEFAULT  NULL;

语句序列很重要,因为一个表不能有两列timestamp类型,其默认值为cureent timestamp


如果您试图将默认值设置为now(),MySQL支持您必须更改该列时间戳的类型而不是日期时间。时间戳默认为当前日期和时间。我认为它将解决您的问题。


您可以解析默认时间戳。首先考虑您正在使用的字符集,例如,如果采用utf8,则此字符集支持所有语言;如果采用laten1,则此字符集仅支持英语。下一个SETP如果您在任何项目下工作,您应该知道客户机时区并选择您是客户机区域。此步骤是必需的。