mysql Alter table设置default的问题,是bug么?
不用不知道,用了没用?
昨天在线上创建了一个表,其中有两个列是timestamp类型的,创建语句假设是这样的:
create table timetest(id int, createtime timestamp,updatetime timestamp);
但是在创建完成之后,显示一下它的创建语句show create table timetest;
CREATE TABLE `timetest` (
`id` int(11) DEFAULT NULL,
`createtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
`updatetime` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4
从上面的结果可以看到,createtime,updatetime多了默认值,第一个默认值为CURRENT_TIMESTAMP,就是说只要修改当前这条记录,那么这个列的值就会被改为now,而第二个列的默认值为'0000-00-00 00:00:00',嗯?怎么回事,完全没有道理!
后面查看了有关文档,发现修改为CURRENT_TIMESTAMP是有原因的,因为很多情况下,一个表中会存在一个修改时间的列,只要这个对象被更新了,那么这个列相应自动修改为当前时间,这是一个比较人性化的针对懒人设置的(源码中可以看到),这个倒是可以理解。
本人之前用数据库比较少,这个特性不太清楚,所以刚好在建表的时候将创建时间放在第一个,将更新时间放在第二个,而据我了解到,自动更新为当前时间的是表中第一个timestamp列,所以刚好将createtime列的设置值设置为CURRENT_TIMESTAMP,并且更新的时候也会更新为CURRENT_TIMESTAMP。
从名字可以看到,这个列是createtime,显然这个值是不会变的,而真正需要这个特性的是updatetime列。
那这个表已经在线上创建了,可能已经在用了,怎么办?总不能删除再创建吧?
然后打算修改一下这个列的default值,语句如下:
alter table timetest alter createtime set default '0000-00-00 00:00:00';
执行后结果会很让人出忽意料,此时再做show create table timetest;
CREATE TABLE `timetest` (
`id` int(11) DEFAULT NULL,
`createtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
`updatetime` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4
嗯?怎么还是一样的?为什么?
看来只能从源码入手了,因为没有任何报错信息,显示的是成功了。
在mysql_alter_table函数中有下面一段代码:
if (!thd->variables.explicit_defaults_for_timestamp)
promote_first_timestamp_column(&alter_info->create_list);
从名字就可以看到,它是将第一个timestamp列的default值提升为CURRENT_TIMESTAMP,函数实现如下:
void promote_first_timestamp_column(List<Create_field> *column_definitions)
{
List_iterator<Create_field> it(*column_definitions);
Create_field *column_definition;
while ((column_definition= it++) != NULL)
{
if (column_definition->sql_type == MYSQL_TYPE_TIMESTAMP || // TIMESTAMP
column_definition->sql_type == MYSQL_TYPE_TIMESTAMP2 || // ms TIMESTAMP
column_definition->unireg_check == Field::TIMESTAMP_OLD_FIELD) // Legacy
{
if ((column_definition->flags & NOT_NULL_FLAG) != 0 && // NOT NULL,
column_definition->def == NULL && // no constant default,
column_definition->unireg_check == Field::NONE) // no function default
{
DBUG_PRINT("info", ("First TIMESTAMP column '%s' was promoted to "
"DEFAULT CURRENT_TIMESTAMP ON UPDATE "
"CURRENT_TIMESTAMP",
column_definition->field_name));
column_definition->unireg_check= Field::TIMESTAMP_DNUN_FIELD;
}
return;
}
}
}
可以看出,如果有一个列为timestamp,且没有指定default值(column_definition->def == NULL),且column_definition->unireg_check == Field::NONE时,就会将column_definition->unireg_check值修改为TIMESTAMP_DNUN_FIELD,说明如果某一个列有这个值,则它表示的就是DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,但是通过跟踪代码发现,这个函数进来之后,根本没有执行这个赋值语句,因为column_definition->unireg_check的值已经是Field::TIMESTAMP_DNUN_FIELD了,为什么已经是这个值呢?因为在修改这个列的时候,新列的unireg_check值是复制的原来列的值,而新列的column_definition->def值是不为空的,它是'0000-00-00 00:00:00',那么最终的结果就是,这个列被修改为unireg_check为TIMESTAMP_DNUN_FIELD,同时def为'0000-00-00 00:00:00'值的列。
那么现在看看更新操作是如何做的
在mysql_update中有下面的2行代码:
if (update.add_function_default_columns(table, table->write_set))
DBUG_RETURN(1);
这里面所做的工作就是将所有的属性为Field::TIMESTAMP_DNUN_FIELD的timestamp类型的列自动加入更新为CURRENT_TIMESTAMP的操作,那么从这里可以看出,只要是有Field::TIMESTAMP_DNUN_FIELD属性的列,那么它这个列就具有了DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP属性。
还可以从show create table timetest;中查看它是如何判断是不是要输出DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP:
在函数print_default_clause中,有如下代码段:
static bool print_default_clause(THD *thd, Field *field, String *def_value,
bool quoted)
{
enum enum_field_types field_type= field->type();
const bool has_now_default= field->has_insert_default_function();
if (has_default)
{
if (has_now_default)
{
def_value->append(STRING_WITH_LEN("CURRENT_TIMESTAMP"));
...
}
...
}
...
}
而函数has_insert_default_function的实现:
bool has_insert_default_function() const
{
return unireg_check == TIMESTAMP_DN_FIELD ||
unireg_check == TIMESTAMP_DNUN_FIELD;
}
这里更加可以坚定的证实它完全是通过TIMESTAMP_DNUN_FIELD来判断的。
那么现在回到上面修改失败的问题,其实还是promote_first_timestamp_column函数的问题,因为里面它只考虑了unireg_check值,而没有考虑是不是这个列的默认值column_definition->def为空,在def及unireg_check之间,应该是def优先级更高的,所以这里应该再做一个降级的,只要def不为空,就需要将unireg_check丢弃,而这里没有做任何处理。
也许这里正是这个问题的结症所在。
那这个问题如果不修改的话有解决办法么?经过应钢同学的指点,它给我如下语句:
语法:MODIFY [COLUMN] col_name column_definition
语句:alter table timetest modify createtime timestamp NOT NULL DEFAULT '0000-00-00 00:00:00';
执行完成之后,再做show create table timetest;
CREATE TABLE `timetest` (
`id` int(11) DEFAULT NULL,
`createtime` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',
`updatetime` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4
哇,修改过来了,但是这个修改方式是将所有的这个列的定义都改了,所以不会继承任何东西。
另一个解决方式:
也许有人已经发现了,上面的代码中有这样的行:
if (!thd->variables.explicit_defaults_for_timestamp)
promote_first_timestamp_column(&alter_info->create_list);
这里有一个条件,就是判断会话参数explicit_defaults_for_timestamp,如果设置为1则不转换,就不会出现第一个timestamp类型的列被自动设置为DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP了,而下面看看这个参数的信息:
/**
This variable is read only to users. It can be enabled or disabled
only at mysqld startup. This variable is used by User thread and
as well as by replication slave applier thread to apply relay_log.
Slave applier thread enables/disables this option based on
relay_log's from replication master versions. There is possibility of
slave applier thread and User thread to have different setting for
explicit_defaults_for_timestamp, hence this options is defined as
SESSION_VAR rather than GLOBAL_VAR.
*/
static Sys_var_mybool Sys_explicit_defaults_for_timestamp(
"explicit_defaults_for_timestamp",
"This option causes CREATE TABLE to create all TIMESTAMP columns "
"as NULL with DEFAULT NULL attribute, Without this option, "
"TIMESTAMP columns are NOT NULL and have implicit DEFAULT clauses. "
"The old behavior is deprecated.",
READ_ONLY SESSION_VAR(explicit_defaults_for_timestamp),
CMD_LINE(OPT_ARG), DEFAULT(FALSE));
默认值为FALSE,并且是一个只读的会话变量,只能通过系统启动时设置这个值。
那么如果设置了explicit_defaults_for_timestamp=1,则不会自动转换。
结论:
alter table timetest alter createtime set default '0000-00-00 00:00:00';
这个方式不起作用,所谓不用不知道,用了没用。
alter table timetest modify createtime timestamp NOT NULL DEFAULT '0000-00-00 00:00:00';
这个方式修改所有的列类型
explicit_defaults_for_timestamp=1
设置了这个完全关闭这个功能。
突然用mysql多了还真现不少问题,继续发现,继续解决。
mysql Alter table设置default的问题,是bug么?的更多相关文章
- MySQL ALTER TABLE: ALTER vs CHANGE vs MODIFY COLUMN
ALTER COLUMN 语法: ALTER [COLUMN] col_name {SET DEFAULT literal | DROP DEFAULT} 作用: 设置或删除列的默认值.该操作会直接修 ...
- mysql ALTER TABLE语句 语法
mysql ALTER TABLE语句 语法 作用:用于在已有的表中添加.修改或删除列.无铁芯直线电机 语法:添加列:ALTER TABLE table_name ADD column_name da ...
- 【待整理】MySQL alter table modify vs alter table add产生state不一样
MySQL:5.6.35 OS:redhat5.8 今天更新数据库某些表字段,有如下两SQL: ①alter table xx modify xxxx;(表大概是77w) ②alter table s ...
- MySQL ALTER TABLE语法
先看一下定义(密密麻麻) ALTER TABLE tbl_name [alter_specification [, alter_specification] ...] [partition_optio ...
- MySQL alter table时执行innobackupex全备再看Seconds_Behind_Master
1.场景描述 早上7:25 接到Report中心同学告警,昨天业务报表数据没有完整跑出来,缺少500位业务员的数据,并且很快定位到,缺少的是huabei_order库上的数据.Report中心的数据是 ...
- mysql alter table
准备: create table t(x int,y int); 用法 1: 修改列的数据类 alter table t modify column y nvarchar(32); 用法2: 给表加一 ...
- EDM排版table设置padding在ie7下bug
今天搞EDM发现一个在ie7下很扯淡的bug,由于以前没遇到过,所以花了点时间来解决下. IE7下bug重现: <table cellpadding="0" cellspac ...
- MySQL字符集的设置
Notice:文章基于ubuntu系统而写 1.关于MySQL字符集 MySQL的字符集支持(Character Set Support)有两个方面: 字符集(Character set)和排序方式( ...
- mysql 字符编码设置
安装mysql时如果字符编码为默认值latin1,则需要修改为utf8以便支持中文数据. 命令如下: 1.显示数据库字符集 mysql> show create database test;+- ...
随机推荐
- 一道JS面试题引发的血案
刚入职新公司,属于公司萌新一枚,一天下午对着屏幕看代码架构时. BI项目组长给我看了一道面试别人的JS面试题. 虽然答对了,但把理由说错了,照样不及格. 话不多说,直接上题: var a = 1; s ...
- Node.js 安装及环境配置之 Windows 篇
一.安装环境 1.本机系统:Windows 10 企业版(64位)2.Node.js:node-v8.9.4-x64.msi(64位) 二.安装Node.js步骤 1.下载对应自己系统对应的 Node ...
- 使用node和express+mongodb实现数据增删改功能
2018即将过去,2019即将来临,前端技术不断在在更新,学的东西越来越多.我们只有不断的学习,才不能被淘汰.在前后端分离的一个时代,后端提供接口,前端调用接口,逻辑判断,每个都是独立的工作.如果自己 ...
- [Python学习笔记-005] 理解yield
网络上介绍yield的文章很多,但大多讲得过于复杂或者追求全面以至于反而不好理解.本文用一个极简的例子给出参考资料[1]中的讲解,因为个人觉得其讲解最为通俗易懂,读者只需要对Python的列表有所了解 ...
- QT QPushButton
#include<QApplication> #include<QWidget> #include<QPushButton> #include<QMenu&g ...
- android开发学习笔记系列(4)--android动态布局
前言 在做一个有关苏果APP的项目中,但是fuck的是,我完全使用相对布局之后及线性布局之后发现坑爹的事情了,屏幕不能适配,这是多大的痛,意味着,必须使用相应的代码实现动态布局!呵呵,不做项目不知道, ...
- 初入Java后端之Servlet
初入Java后端之Servlet 后端 Servlet 什么是Servlet? Servlet实际上是一个按照Servlet规范写的Java类.是运行在Web服务端的Java应用程序.与Java程序 ...
- Docker基础-镜像操作
1.搜寻镜像 docker search [TERM] 可以搜索远程仓库中共享的镜像,默认搜索官方仓库的镜像. docker search参数: --automated=true|false :仅显示 ...
- StreamReader 的ReadLine,Read,ReadToEnd方法
1.ReadLine() 当遇到\n \r 或者是\r\n的时候 此方法返回这前面的字符串,然后内部的指针往后移一位下次从新的地方开始读,直到遇到数据的结尾处返回null,所以经常这样使用 ...
- 让 VS2010 支持 HTML5 和 CSS3.0
现在的热门话题之一是HTML5 和 CSS3.好的, 它们都很时髦,它们也必然会影响网络开发的未来. 让我们尝尝鲜,花点时间安装设置一下,尽快让Visual Studio2010支持HTML5 和 C ...