1

我有一张有 100 万条记录的表。我需要能够将这些记录移动到另一个数据库和另一个表。

我正在使用存储过程来获取数据。它填充数据适配器,然后将数据 bcp 到新表中。

我们使用的是 SQL Server 2005 和 C# 4。我们将使用 C# 4.6 或 5.0 迁移到 SQL Server 2012 或 2014 和 Visual Studio 2015。如果有任何功能可以使它正常工作。

  • 对于 10k 条记录,该过程不到 1 秒
  • 对于 500k 条记录,dataadapter 内存不足,进程失败。批处理到 10 万条记录,选择语句是 SQL 中的问题,一次返回 10 万条记录,每个循环需要 2 分钟。

有没有办法,或者我下面的代码有什么问题,可以防止数据适配器被填充,而是映射列并BulkCopy保持服务器端,只是将记录从数据库推送到新表,就像 SSIS 一样?

大容量复制本身似乎快如闪电,但适配器填充失败,因为它试图用 100 万条记录填充适配器时内存不足。一次不做 1 行,我只想在表之间移动数据。

一个表有 27 列,其中 5 列不在表 2 中,表 2 有 32 列,并且两个表中的某些列的名称不同。

这是概念证明 (PoC)。

sourceConn_transDB.Open();
SqlCommand sourceCommand = new SqlCommand(queryString, sourceConn_transDB);
DataTable table = new DataTable();

sourceCommand.CommandTimeout = 600;

using (var adapter = new SqlDataAdapter(sourceCommand))
{
    WriteLineWithTime("Adapter Fill");
    adapter.Fill(table);
}

if ((table == null) || (table.Rows.Count <= 0))
   break;

using (SqlBulkCopy bulk = new SqlBulkCopy(targetConn_reportDB, SqlBulkCopyOptions.KeepIdentity, null) { DestinationTableName = "PatientEvent" })
{
    bulk.ColumnMappings.Add(new SqlBulkCopyColumnMapping("PatientID", "PatientID"));
}
4

1 回答 1

5

您是否尝试过使用WriteToServer带 a 的重载IDataReader,那么您根本不需要使用 a DataTable

sourceConn_transDB.Open();
using(SqlCommand sourceCommand = new SqlCommand(queryString, sourceConn_transDB))
{
    sourceCommand.CommandTimeout = 600;

    using (SqlDataReader reader = sourceCommand.ExecuteReader())
    using (SqlBulkCopy bulk = new SqlBulkCopy(targetConn_reportDB, SqlBulkCopyOptions.KeepIdentity, null) { DestinationTableName = "PatientEvent" })
    {
        bulk.ColumnMappings.Add(new SqlBulkCopyColumnMapping("PatientID", "PatientID"));
        bulk.WriteToServer(reader);
    }
}
于 2016-01-07T18:01:44.887 回答