我试着做这个查询

INSERT INTO dbo.tbl_A_archive
  SELECT *
  FROM SERVER0031.DB.dbo.tbl_A

但即使在我跑了之后

set identity_insert dbo.tbl_A_archive on

我得到这个错误消息

表'dbo中标识列的显式值。tbl_A_archive'只能在使用列列表且IDENTITY_INSERT为ON时指定。

tbl_A是一个行和宽都很大的表,也就是说它有很多列。我不想手动输入所有的列。我怎样才能让它工作呢?


当前回答

有一个或多个列具有自动递增属性,或者该属性的值将作为约束计算。你要修改这一列。

有两种解决方法 1)显式地提到其他列并只设置它们的值,PrimaryKey或自动递增列值将自动设置。

2)您可以打开IDENTITY_INSERT,然后执行插入查询,最后关闭IDENTITY_INSERT。

建议:按照第一步进行,因为这是更合适和有效的方法。

要了解更多信息,请阅读关于SQL-helper的文章。

其他回答

有一个或多个列具有自动递增属性,或者该属性的值将作为约束计算。你要修改这一列。

有两种解决方法 1)显式地提到其他列并只设置它们的值,PrimaryKey或自动递增列值将自动设置。

2)您可以打开IDENTITY_INSERT,然后执行插入查询,最后关闭IDENTITY_INSERT。

建议:按照第一步进行,因为这是更合适和有效的方法。

要了解更多信息,请阅读关于SQL-helper的文章。

总结

SQL Server不允许您在标识列中插入显式值,除非您使用列列表。因此,你有以下选项:

制作一个列列表(手动或使用工具,见下文)

OR

将tbl_A_archive中的标识列设置为常规的非标识列:如果您的表是一个归档表,并且总是为标识列显式指定一个值,那么为什么还需要标识列呢?只需要使用常规int型就可以了。


方案一详细说明

而不是

SET IDENTITY_INSERT archive_table ON;

INSERT INTO archive_table
  SELECT *
  FROM source_table;

SET IDENTITY_INSERT archive_table OFF;

你需要写

SET IDENTITY_INSERT archive_table ON;

INSERT INTO archive_table (field1, field2, ...)
  SELECT field1, field2, ...
  FROM source_table;

SET IDENTITY_INSERT archive_table OFF;

用field1, field2,…包含表中所有列的名称。如果您想自动生成列列表,请查看Dave的答案或Andomar的答案。


方案二详细说明

不幸的是,仅仅将单位int列的“类型”更改为非单位int列是不可能的。基本上,你有以下几种选择:

如果存档表还不包含数据,则删除该列并添加一个不带标识的新列。

OR

使用SQL Server Management Studio将存档表中的标识列的标识规范/(是标识)属性设置为No。在幕后,这将创建一个脚本来重新创建表并复制现有数据,因此,要做到这一点,您还需要取消设置Tools/Options/Designers/ table和Database Designers/Prevent保存需要重新创建表的更改。

OR

使用此回答中描述的解决方法之一:从表中的列中删除标识

如果“归档”表是您的主表的精确副本,那么我只是建议您删除id是标识列的事实。这样你就可以插入它们了。

或者,您可以使用以下语句允许和禁止对表进行标识插入

SET IDENTITY_INSERT tbl_A_archive ON
--Your inserts here
SET IDENTITY_INSERT tbl_A_archive OFF

最后,如果您需要标识列按原样工作,那么您总是可以运行存储的过程。

sp_columns tbl_A_archive 

这将返回表中的所有列,然后您可以将其剪切并粘贴到您的查询中。(这几乎总是比使用*更好)

如果您正在使用SQL Server Management Studio,您不必自己键入列列表-只需在对象资源管理器中右键单击表,并选择脚本表作为-> SELECT到->新建查询编辑器窗口。

如果你不是,那么类似的查询应该有助于作为一个起点:

SELECT SUBSTRING(
    (SELECT ', ' + QUOTENAME(COLUMN_NAME)
        FROM INFORMATION_SCHEMA.COLUMNS
        WHERE TABLE_NAME = 'tbl_A'
        ORDER BY ORDINAL_POSITION
        FOR XML path('')),
    3,
    200000);

同意海因茨的回答。对于第一个第二个选项,下面是一个查询,它在表中生成一个逗号分隔的列列表:

select name + ', ' as [text()] 
from sys.columns 
where object_id = object_id('YourTable') 
for xml path('')

对于大的表格,这可以节省大量的打字工作:)