我正在使用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;
答案 0 :(得分:11)
要回答我自己的问题,我找不到更简单的方法来做到这一点。我最终删除了所有外键约束,更改字段类型,然后再添加所有外键约束。
正如R. Bemrose所说,使用SET foreign_key_checks = 0;
只会在添加或更改数据时有所帮助,但不允许ALTER TABLE
命令破坏外键约束。
答案 1 :(得分:10)
这是我对这个帖子的小小贡献。感谢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然后执行查询。
答案 2 :(得分: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。我建议 - 如果可能的话 - 在此阶段阻止来自任何客户的连接,以最大限度地减少意外约束违规的风险。
至于密钥本身:当你更改了表数据类型时,它们也需要被删除和重新创建。
答案 3 :(得分:1)
您可以通过键入
暂时禁用外键SET foreign_key_checks = 0;
并重新启用它们
SET foreign_key_checks = 1;
我认为这需要管理员权限,因为它旨在将数据导入数据库。
编辑:作为对编辑的反应,看起来它只会禁用DML语句(插入,更新,删除)的约束,但不会禁用DDL语句(alter table,drop table等... )。
答案 4 :(得分:1)
如果你可以停止数据库,那么尝试将表转储到文本文件,在文件中手动更改列定义并重新导入表。
答案 5 :(得分: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`
答案 6 :(得分:0)
请不要使用上面的答案
我的解决方案,处理可空值、默认值、自动增量以及注释:
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;