MySQL更改外键类型

Posted

技术标签:

【中文标题】MySQL更改外键类型【英文标题】:MySQL change type of foreign key 【发布时间】:2009-05-06 14:37:31 【问题描述】:

我正在使用 mysql,并且我有一个带有索引的表,该索引在许多其他表中用作外键。我想更改索引的数据类型(从有符号整数到无符号整数),最好的方法是什么?

我尝试更改索引字段的数据类型,但失败了,因为它被用作其他表的外键。我尝试更改其中一个外键的数据类型,但失败了,因为它与索引的数据类型不匹配。

我想我可以手动删除所有外键约束,更改数据类型并重新添加约束,但这将是很多工作,因为我有很多表使用这个索引作为外键。有没有办法在进行更改时暂时关闭外键约束?另外,有没有办法获取引用索引作为外键的所有字段的列表?

更新: 我尝试在关闭外键检查后修改一个外键,但似乎并没有关闭检查:

SET foreign_key_checks = 0;

ALTER TABLE `escolaterrafir`.`t23_aluno` MODIFY COLUMN `a21_saida_id` INTEGER DEFAULT NULL;

这是错误:

------------------------
LATEST FOREIGN KEY ERROR
------------------------
090506 11:57:34 Error in foreign key constraint of table escolaterrafir/t23_aluno:
there is no index in the table which would contain
the columns as the first columns, or the data types in the
table do not match to the ones in the referenced table
or one of the ON ... SET NULL columns is declared NOT NULL. Constraint:
,
  CONSTRAINT FK_t23_aluno_8 FOREIGN KEY (a21_saida_id) REFERENCES t21_turma (A21_ID)

索引表的定义:

DROP TABLE IF EXISTS `escolaterrafir`.`t21_turma`;
CREATE TABLE  `escolaterrafir`.`t21_turma` (
  `A21_ID` int(10) unsigned NOT NULL auto_increment,
  ...
) ENGINE=InnoDB AUTO_INCREMENT=51 DEFAULT CHARSET=latin1;

以及具有指向它的外键的表:

DROP TABLE IF EXISTS `escolaterrafir`.`t23_aluno`;
CREATE TABLE  `escolaterrafir`.`t23_aluno` (
  ...
  `a21_saida_id` int(10) unsigned default NULL,
  ...
  KEY `Index_7` (`a23_id_pedagogica`),
  ...
  CONSTRAINT `FK_t23_aluno_8` FOREIGN KEY (`a21_saida_id`) REFERENCES `t21_turma` (`A21_ID`)
) ENGINE=InnoDB AUTO_INCREMENT=387 DEFAULT CHARSET=latin1;

【问题讨论】:

【参考方案1】:

这是我对这个线程的小贡献。感谢 Daniel Schneller 的启发并为我提供了很大一部分解决方案!

set group_concat_max_len = 2048;
set @table_name = "YourTableName";
set @change = "bigint unsigned";
select distinct table_name,
       column_name,
       constraint_name,
       referenced_table_name,
       referenced_column_name,
       CONCAT(
           GROUP_CONCAT('ALTER TABLE ',table_name,' DROP FOREIGN KEY ',constraint_name SEPARATOR ';'),
           ';',
           GROUP_CONCAT('ALTER TABLE `',table_name,'` CHANGE `',column_name,'` `',column_name,'` ',@change SEPARATOR ';'),
           ';',
           CONCAT('ALTER TABLE `',@table_name,'` CHANGE `',referenced_column_name,'` `',referenced_column_name,'` ',@change),
           ';',
           GROUP_CONCAT('ALTER TABLE `',table_name,'` ADD CONSTRAINT `',constraint_name,'` FOREIGN KEY(',column_name,') REFERENCES ',referenced_table_name,'(',referenced_column_name,')' SEPARATOR ';')
       ) as query
from   INFORMATION_SCHEMA.key_column_usage
where  referenced_table_name is not null
   and referenced_column_name is not null
   and referenced_table_name = @table_name
group by referenced_table_name

通过设置@table_name 和@change,您可以生成查询。 @table_name 应该是具有主键的表的表名(它将查找使用该列作为外键的表)并将其类型更改为@change。

我不得不像这样换几张桌子,这样效果很好。我只需要更改@table_name 然后执行查询。

【讨论】:

您应该提到您正在使用INFORMATION_SCHEMA 数据库来进行这些更改,该数据库包含所有常规数据库的定义。我认为您还应该在 where 子句中添加and TABLE_SCHEMA = my_database_name,否则您的查询将影响所有数据库中的同名表,其中“my_database_name”将是您的表所在的数据库的名称。 附加到我之前的评论,也许CONSTRAINT_SCHEMA 而不是TABLE_SCHEMA 是你需要做的事情?? 必须替换:CONCAT('ALTER TABLE `',@table_name,'` CHANGE `id` `id` ',@change),CONCAT('ALTER TABLE `',@table_name,'` CHANGE `',column_name,'` `',column_name,'` ',@change),from key_column_usagefrom INFORMATION_SCHEMA.KEY_COLUMN_USAGE 小心。这也会删除其他列定义,例如 AUTO_INCREMENT。 这是巨大的,我正在为我们的一个表获取超过 1000 行的 SQL,并且在自己接近迁移这些列之前可能已经失去了所有的头发。谢谢!【参考方案2】:

要回答我自己的问题,我找不到更简单的方法。我最终删除了所有外键约束,更改了字段类型,然后添加了所有外键约束。

正如 R. Bemrose 所指出的,使用 SET foreign_key_checks = 0; 仅在添加或更改数据时有帮助,但不允许使用会破坏外键约束的 ALTER TABLE 命令。

【讨论】:

我遇到了同样的问题,为了解决它我:mysqldump'ed 数据库到一个文本文件,更改了相关的列并重新导入它。我认为这是最简单/最快的方法。【参考方案3】:

要了解外键约束的使用,请在 INFORMATION_SCHEMA 数据库上发出以下查询:

select distinct table_name, 
       column_name, 
       constraint_name,  
       referenced_table_name, 
       referenced_column_name 
from   key_column_usage 
where  constraint_schema = 'XXX' 
   and referenced_table_name is not null 
   and referenced_column_name is not null;

XXX 替换为您的架构名称。这将为您提供将其他列作为外键引用的表和列的列表。

不幸的是,架构更改是非事务性的,所以我担心您确实必须暂时禁用该操作的 foreign_key_checks。我建议 - 如果可能的话 - 在此阶段阻止来自任何客户端的连接,以尽量减少意外违反约束的风险。

至于键本身:当您更改表数据类型时,它们也需要被删除和重新创建。

【讨论】:

【参考方案4】:

如果您可以停止数据库,请尝试将表转储到文本文件,在文件中手动更改列定义并重新导入表。

【讨论】:

【参考方案5】:

您可以通过键入暂时禁用外键

SET foreign_key_checks = 0;

并重新启用它们

SET foreign_key_checks = 1;

我认为这需要管理员权限,因为它旨在将数据导入数据库。

编辑:作为对您的编辑的反应,看起来它只禁用了 DML 语句(插入、更新、删除)的约束,而不禁用 DDL 语句(更改表、删除表等... )。

【讨论】:

【参考方案6】:

这也是我对这个线程的小贡献。感谢 Daniel Schneller 和 Wiktor Jarka 的启发,并为我提供了很大一部分解决方案! 此解决方案修复了数据库和自动增量问题。

SET group_concat_max_len = 2048;
SET @database_name = "my_database";
SET @table_name = "my_table";
SET @change = "tinyint unsigned";
SELECT DISTINCT 
       `table_name`,
       `column_name`,
       `constraint_name`,
       `referenced_table_name`,
       `referenced_column_name`,
       CONCAT(
           GROUP_CONCAT('ALTER TABLE `',table_name,'` DROP FOREIGN KEY `',constraint_name, '`' SEPARATOR ';'),
           ';',
           GROUP_CONCAT('ALTER TABLE `',table_name,'` CHANGE `',column_name,'` `',column_name,'` ',@change SEPARATOR ';'),
           ';',
           CONCAT('ALTER TABLE `',@table_name,'` CHANGE `',referenced_column_name,'` `',referenced_column_name,'` ',@change, ' NOT NULL AUTO_INCREMENT'),
           ';',
           GROUP_CONCAT('ALTER TABLE `',table_name,'` ADD CONSTRAINT `',constraint_name,'` FOREIGN KEY(',column_name,') REFERENCES ',referenced_table_name,'(',referenced_column_name,')' SEPARATOR ';'), 
           ';'
       ) AS query
FROM   `information_schema`.`key_column_usage`
WHERE  `referenced_table_name` IS NOT NULL
   AND `referenced_column_name` IS NOT NULL
   AND `constraint_schema` = @database_name
   AND `referenced_table_name` = @table_name
GROUP BY `referenced_table_name`

【讨论】:

【参考方案7】:

请不要使用上面的答案

    组 concat 是个坏主意,因为您无法控制输出的长度 您应该使用 MODIFY 而不是 CHANGE,因为您不会重命名列 其他答案不处理 null、默认值,因此可能会弄乱您的数据库

我的解决方案,处理可空值、默认值、自动增量以及 cmets:

SET @table_schema = "database";
SET @table_name = "table";
SET @column_name = "column";
SET @new_type = "bigint unsigned";


SELECT CONCAT('ALTER TABLE `', table_schema,'`.`', table_name, '` DROP FOREIGN KEY `', constraint_name,  '`;' ) AS drop_fk
FROM information_schema.key_column_usage
WHERE referenced_table_schema =  @table_schema AND referenced_table_name = @table_name and referenced_column_name = @column_name;


SELECT CONCAT('ALTER TABLE `', kcu.table_schema,'`.`', kcu.table_name,'` MODIFY `', kcu.column_name, '` ',
@new_type, 
if(c.is_nullable = 'NO', ' NOT NULL', ' NULL'),
if(c.column_default is not null, CONCAT(' DEFAULT ', QUOTE(c.column_default)), ''),
if(c.extra <> '', CONCAT(' ', c.extra), ''),
if(c.column_comment <> '', CONCAT(' COMMENT ', QUOTE(c.column_comment), ''), ''),
';') AS modify
FROM information_schema.key_column_usage AS kcu
JOIN information_schema.columns AS c ON c.table_schema=kcu.table_schema AND c.table_name=kcu.table_name AND c.column_name=kcu.column_name
WHERE kcu.referenced_table_schema = @table_schema AND kcu.referenced_table_name = @table_name AND kcu.referenced_column_name = @column_name;


SELECT CONCAT('ALTER TABLE `', table_schema,'`.`',table_name,'` ADD CONSTRAINT `',constraint_name ,'` FOREIGN KEY (`', column_name, '`) REFERENCES `',
referenced_table_schema ,'`.`', referenced_table_name , '` (`', referenced_column_name,'`);') AS create_fk
FROM information_schema.key_column_usage
WHERE referenced_table_schema = @table_schema AND referenced_table_name = @table_name AND referenced_column_name = @column_name;

【讨论】:

我忘记了您可能必须在更改索引的列类型后运行 OPTIMIZE TABLE。记得前阵子换了一个类型,没有重新计算索引,导致性能不好。我不知道mysql是否纠正了这个。

以上是关于MySQL更改外键类型的主要内容,如果未能解决你的问题,请参考以下文章

Sql Server2008数据类型更改不能保存?

MYSQL添加外键关联

实体框架中与代码优先外键的更改冲突

mysql笔记1

mysql怎么更改字段名

如何使用 Laravel 模式构建器更改列类型?