是否可以在MySQL中创建一个UNIX_TIMESTAMP默认列?

Kit*_*nde 35 mysql sql database database-design database-schema

我试图这样做,但似乎MySQL不允许我这样做.是否有解决此问题的方法,或者我希望在我的INSERT查询中始终包含该函数?

CREATE TABLE foo(
  created INT NOT NULL DEFAULT UNIX_TIMESTAMP()
)
Run Code Online (Sandbox Code Playgroud)

我知道接受CURRENT_TIMESTAMP默认值的TIMESTAMP类型,但我的客户端坚持在数据库中使用纪元时间.

Ike*_*ker 50

MySQL实现TIMESTAMP数据类型的方式实际上是将纪元时间存储在数据库中.所以你可以使用一个TIMESTAMP默认值为的列,如果要将其显示为int,则CURRENT_TIMESTAMP应用UNIX_TIMESTAMP()它:

CREATE TABLE foo(
  created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP
);

insert into foo values (current_Date()),(now());

select unix_timestamp(created) from foo;
+-------------------------+
| unix_timestamp(created) |
+-------------------------+
|              1300248000 |
|              1300306959 |
+-------------------------+
2 rows in set (0.00 sec)
Run Code Online (Sandbox Code Playgroud)

但是,如果您真的想要列的数据类型INT,可以使用R. Bemrose的建议并通过触发器设置它:

CREATE TABLE foo(
  created INT NULL
);

delimiter $$

create trigger tr_b_ins_foo before insert on foo for each row
begin
  if (new.created is null)
  then
    set new.created = unix_timestamp();
  end if;
end $$

delimiter ;


insert into foo values (unix_timestamp(current_Date())), (null);

select created from foo;
+------------+
| created    |
+------------+
| 1300248000 |
| 1300306995 |
+------------+
2 rows in set (0.00 sec)
Run Code Online (Sandbox Code Playgroud)

  • 任何选择数据类型的人都应该记住,TIMESTAMP面向的是相同的2038但是面向INT.考虑存储在UNSIGNED INT(11)中的UNIX_TIMESTAMP(Epoch)的原因是将溢出错误推到下个世纪(BIGINT更进一步).当新技术出现时,您的代码很可能需要重写.然而,2038年将在我们的许多生命中发生.我们今天构建的数据库仍然可以作为遗留系统.@JRun - 我希望MySQL中的TIMESTAMP在后端被视为INT,因为它具有相同的数字限制. (2认同)

Joe*_*lli 19

文档:

除了一个例外,默认值必须是常量; 它不能是一个功能或表达.这意味着,例如,您不能将日期列的默认值设置为函数的值,例如NOW()或CURRENT_DATE.例外情况是您可以将CURRENT_TIMESTAMP指定为TIMESTAMP列的默认值.


Arj*_*Raj 8

您可以为此创建触发器.

用于插入

  • 询问

CREATE TRIGGER {trigger_name} BEFORE INSERT ON {table_name} FOR EACH ROW SET new.{field_name} = UNIX_TIMESTAMP(NOW());

  • 在这种情况下

CREATE TRIGGER my_trigger_name_1 BEFORE INSERT ON foo FOR EACH ROW SET new.created = UNIX_TIMESTAMP(NOW());

更新

  • 询问

CREATE TRIGGER {trigger_name} BEFORE UPDATE ON {table_name} FOR EACH ROW SET new.{field_name} = UNIX_TIMESTAMP(NOW());

  • 在这种情况下

CREATE TRIGGER my_trigger_name_2 BEFORE UPDATE ON foo FOR EACH ROW SET new.created = UNIX_TIMESTAMP(NOW());

注意:我不知道MYSQL TRIGGER的性能

请仔细阅读这些链接

  1. 找出实现sql server触发器的一些缺点

  2. 使用触发器

  • _tigers_? - 他们被驯服了,我希望:-) (3认同)

小智 8

正如ktretyak 的回答,使用 MySQL v8+,您可以使用括号来做到这一点。

CREATE TABLE foo(
  created INT NOT NULL DEFAULT (UNIX_TIMESTAMP())
)
Run Code Online (Sandbox Code Playgroud)

经过一些插入后,您可以看到它工作正常。
(在 MySQL 8.0.26 上测试 + 仅文本颜色修补)

测试案例

或者,您也可以使用表达式。
检查ktretyak 的答案

我写这个答案而不是评论或编辑答案(尝试并拒绝),
因为缺乏声誉,并且使用括号现在是一个很好的解决方案。


Pow*_*ord 5

好吧,如果 MySQL 不允许您直接执行此操作,您始终可以使用BEFORE INSERT... FOR EACH ROW触发器来执行此操作。