2

现在我们已经用完了intPK 列(这是一个IDENTITY)的容量,我想这样做bigint,但简单ALTER TABLE似乎无法处理那么大的表。所以我的问题是:如何在保持实际值不变的情况下更改 PK 列的类型,我是否还需要更改引用表?

4

4 回答 4

4

除了 KLE 的建议之外,以下查询可能会有所帮助:

要禁用引用 oldTable 的表的所有约束,请尝试执行以下查询的输出:

SELECT 'ALTER TABLE ' + OBJECT_NAME(fk.parent_object_id) + ' NOCHECK CONSTRAINT ' + fk.name
FROM sys.foreign_keys fk
INNER JOIN sys.foreign_key_columns AS fkc ON fk.OBJECT_ID = fkc.constraint_object_id
WHERE OBJECT_NAME (fk.referenced_object_id) = 'oldTable'

要将所有数据移动到新表中,并更改字段,请尝试以下操作:

INSERT INTO newTable
SELECT CONVERT(BIGINT, ID) AS ID, COL1, COL2, ..., COLN
FROM oldTable

要删除旧表:

DROP TABLE oldTable

要将新表重命名为旧名称:

sp_rename newTable, oldTable

要重新启用引用 oldTable 的表的所有约束,请尝试执行以下查询的输出:

SELECT 'ALTER TABLE ' + OBJECT_NAME(fk.parent_object_id) + ' CHECK CONSTRAINT ' + fk.name
FROM sys.foreign_keys fk
INNER JOIN sys.foreign_key_columns AS fkc ON fk.OBJECT_ID = fkc.constraint_object_id
WHERE OBJECT_NAME (fk.referenced_object_id) = 'oldTable'

希望能帮助到你...

于 2009-09-14T09:52:28.513 回答
3

我们要做的是:

保存你的表

  1. 创建具有正确结构的新表

  2. 禁用这些表以及引用它们的所有约束

  3. 将所有数据移动到新表中,并更改字段;可以分批完成

  4. 空时删除旧表

  5. 将新表重命名为旧名称

  6. 在所有表上启用所有约束(一些 FK 列和约束可能也需要修复......但它们不是 PK,因此它们是可修改的)

    6 编辑(感谢 Alexey)

这很干净,可以分批使用,很好理解。

于 2009-09-14T07:54:02.603 回答
0

您还需要更改子表。毕竟,您现在也将尝试在其中插入一个大整数。我会先改变子表

这不是一个简单或短暂的过程。我建议您在设定的日期告诉您的用户数据库将因维护而停机(您可以衡量开发需要多长时间),并在您制作时将数据库重置为单用户模式这些变化。您不希望在切换到另一个表时丢失用户添加(或更改)到另一个表的数据。如果由于某种原因你不能有一个维护窗口(为了数据完整性,我强烈建议你这样做),那么你必须首先更改子表以避免插入错误,如果你真的接近限制并且将是几乎立即看到大量数字。

确保为整个数据库结构编写脚本,包括默认值、触发器、检查常量索引等,因为您需要重新创建所有内容。

确保通过 dev 上的脚本完成所有这些工作。一旦你测试了这个过程,这将使做一个 prod 变得更加容易。

于 2009-09-14T17:23:25.493 回答
-1

我认为您只能创建一个更改了 PK 数据类型的新数据库,然后导出/导入数据,或批量插入新数据库,然后重命名新数据库。当然,如果您有许多引用的表并且您的新 PK 数据类型与以前的不兼容,那么这是实际的。

于 2009-09-14T07:53:48.953 回答