17

我正在使用 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;
4

7 回答 7

18

这是我对这个线程的小贡献。感谢 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 然后执行查询。

于 2011-02-22T12:04:37.777 回答
14

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

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

于 2009-05-12T17:05:27.703 回答
2

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

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

于 2009-05-06T15:04:12.067 回答
2

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

于 2010-02-03T10:27:36.277 回答
1

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

SET foreign_key_checks = 0;

并重新启用它们

SET foreign_key_checks = 1;

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

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

于 2009-05-06T14:48:15.443 回答
1

这也是我对这个线程的小贡献。感谢 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`
于 2020-05-08T18:28:33.693 回答
0

请不要使用上面的答案

  1. group concat 是个坏主意,因为您不控制输出的长度
  2. 你应该使用 MODIFY 而不是 CHANGE 因为你不重命名列
  3. 其他答案不处理空值,默认值,因此可能会弄乱您的数据库

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

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;
于 2021-04-09T14:37:50.687 回答