我正在使用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;
有帮助吗?

解决方案 2

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

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

其他提示

这是我对这个帖子的小小贡献。感谢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 数据库上发出以下查询:

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。我建议 - 如果可能的话 - 在此阶段阻止来自任何客户的连接,以最大限度地减少意外违反约束的风险。

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

您可以通过键入

暂时禁用外键
SET foreign_key_checks = 0;

并重新启用它们

SET foreign_key_checks = 1;

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

编辑:作为对编辑的反应,看起来它只会禁用DML语句(插入,更新,删除)的约束,但不会禁用DDL语句(alter table,drop table等... )。

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

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top