MS SQL Server(Ver 2008或更高版本) - 如何根据我的ID列

时间:2016-11-12 19:58:08

标签: sql sql-server bulkinsert

我在csv文件中有一个表(ID为numeric)。

我手动将文件中的信息上传到SQL Server数据表(将我的ID列创建为numeric)。

但是,我想将我的ID列重新创建为自动编号ID列,以使用最新条目继续编号。

示例:该表的ID为1,5,10。我想重新创建自动增量(IDENTITY)ID列(留下我的旧ID' s),下一行插入继续ID 11。

我认为没有一种方法可以实现这一目标。但我想知道我应该遵循的步骤。

2 个答案:

答案 0 :(得分:2)

这是一个脚本,可以让您了解一种方法。

IF OBJECT_ID('DELETEME.dbo.Tbl') IS NOT NULL
    BEGIN
        DROP TABLE Tbl
    END

IF OBJECT_ID('DELETEME.dbo.stageTbl') IS NOT NULL
    BEGIN
        DROP TABLE stageTbl
    END

CREATE TABLE Tbl (
    ID INT
    ,A CHAR(1)
)

INSERT INTO Tbl VALUES (1,'A'),(2,'B'),(10,'C')

SELECT *
FROM
    Tbl

EXEC sp_rename 'DELETEME.dbo.Tbl', 'stageTbl', 'OBJECT'
--renames original table

--create script for the new table
CREATE TABLE Tbl (
    ID INT NOT NULL IDENTITY(1,1)
    ,A CHAR(1)
)


--have to set IDENTITY_INSERT on to insert the ID into an IDENTITY column
SET IDENTITY_INSERT Tbl ON

INSERT INTO Tbl (ID, A)
SELECT ID, A
FROM
    stageTbl

SET IDENTITY_INSERT Tbl OFF

DROP TABLE stageTbl
--drops original table

DBCC CHECKIDENT('Tbl', RESEED, 222)
--sets the number you want to with next if you set as 222 the next identity will be 223

INSERT INTO Tbl (A) VALUES ('D')

SELECT *
FROM
    Tbl

基本步骤

  • 重命名原始表格(如果您希望新表格与旧表格相同,我希望首先重命名,因为新表格上有自动生成的约束名称等)
  • 使用列作为标识列创建新表
  • 启用IDENTITY_INSERT
  • 从旧表中选择所有记录到新表
  • 关闭IDENTITY_INSERT
  • 你不必这样做,但你可以使用你想要的任何数字来启用身份,否则SQL服务器会根据最大的ID值自动执行此操作。
  • 删除重命名的原始表

答案 1 :(得分:0)

感谢Matt帮我解决原始问题。 我想分享一个C#方法,用于自动化所有必要的步骤:

- 免责声明:使用this是我的类,它与MS SQL Server连接,用于读取SELECT语句(并返回一个DataTable)和执行SQL查询等。希望有人能找到这段代码有帮助的(AS-IS) -

/// <summary> Recreate an ID with auto-incremental when the table has the ID without this option.
/// <para>Automatically will rename the original table to TABLENAME_TO_DELETE (The process will require copy and recreate the table, then the process will duplicate the information) </para></summary>
/// <param name="strTable">SQL table</param>
/// <param name="strId">ID column</param>
public string recreateIdentityColumn(string strTable, string strId)
{
    string strLog = "Table: {0} - ID: {1}".fwFormat(strTable, strId);

    string strNewTable = strTable + "_" + fw.rnd(1, 1000).ToString() + fw.rnd(5000, 10000);
    DataTable dtTable = this.fillDataTable("SELECT COLUMN_NAME, DATA_TYPE, NUMERIC_PRECISION, NUMERIC_SCALE " +
                                            "FROM Information_SCHEMA.COLUMNS " +
                                            "WHERE TABLE_NAME = '" + strTable + "'");

    if (!dtTable.fwHasData()) throw new Exception("The current table '" + strTable + "' doesn't exists");
    DataRow[] drIdInfo = dtTable.Select("COLUMN_NAME = '" + strId + "'");

    if (!drIdInfo.fwHasData()) throw new Exception("The ID column '" + strId + "' doesn't exists in the table '" + strTable + "'");

    string strIdType = "";
    string strColumns = "";

    strIdType = drIdInfo[0]["DATA_TYPE"].fwEmpty("");

    if (strIdType.fwContains("decimal"))
        strIdType += "({0}, {1})".fwFormat(drIdInfo[0]["NUMERIC_PRECISION"].ToString(), drIdInfo[0]["NUMERIC_SCALE"].ToString());

    strLog += "\r\nID DataType: " + strIdType;

    foreach (DataRow drInfo in dtTable.Rows)
        strColumns += ",[" + drInfo["COLUMN_NAME"].ToString() + "]";

    strId = "[" + strId.TrimStart('[').TrimEnd(']') + "]";
    strColumns = strColumns.TrimStart(',');
    strLog += "\r\nColumns: " + strColumns;

    try
    {
        // Rule 1: Clone the table (Only the structure)
        this.executeQuery("SELECT TOP 0 * INTO " + strNewTable + " FROM " + strTable);

        // Rule 2: Remove the ID from the clone table
        this.executeQuery("ALTER TABLE " + strNewTable + " DROP COLUMN " + strId);

        // Rule 3: Add the ID column with the identity property
        this.executeQuery("ALTER TABLE " + strNewTable + " ADD " + strId + " " + strIdType + " IDENTITY(1,1)");

        // Rule 4: Allow manual insertion of ID in the identity column
        this.executeQuery("SET IDENTITY_INSERT " + strNewTable + " ON");

        // Rule 5: Copy the rows into the table
        int intTotalRows = this.rowCount(strTable);
        int intTotalNewRows = this.executeQuery("INSERT INTO " + strNewTable + "(" + strColumns + ") " +
                            "SELECT " + strColumns + " FROM " + strTable);

        strLog += "\r\nOriginal rows {0} - New rows {1}".fwFormat(intTotalRows.ToString(), intTotalNewRows.ToString());

        // Rule 6: Return the insertion of identity rows to a normal state
        this.executeQuery("SET IDENTITY_INSERT " + strNewTable + " OFF");

        // Rule 7: Rename the table with NO IDENTITY as OLD and rename the table with INDENTITY ID as NEW/ACTUAL
        this.executeQuery("EXEC sp_rename '" + strTable + "', '" + strTable + "_TO_DELETE', 'OBJECT'");
        this.executeQuery("EXEC sp_rename '" + strNewTable + "', '" + strTable + "', 'OBJECT'");
        strLog += "\r\nProcess run without problems";
        return strLog;
    }
    catch (Exception ex)
    {
        strLog += "\r\nException occur";
        throw ex;
    }
}