Каков наилучший способ массовой вставки базы данных из C #? - PullRequest
29 голосов
/ 25 марта 2009

Как мне / как лучше всего выполнять массовые вставки в базу данных?

В C # я перебираю коллекцию и вызываю хранимую процедуру вставки для каждого элемента в коллекции.

Как отправить все данные за один вызов базы данных?

например. скажем, у меня есть список людей (List<Person>), содержащий 10 пунктов. В настоящее время я вызываю сохраненный процесс InsertPerson 10 раз. Я хотел бы уменьшить это до 1 звонка.

Я использую MS SQL Server 2005.

Ответы [ 10 ]

27 голосов
/ 31 марта 2010

CsharperGuyInLondon, вот простой пример кода SqlBulkCopy:

using System.Data.SqlClient;

DataTable table = new DataTable("States");
// construct DataTable
table.Columns.Add(new DataColumn("id_state", typeof(int))); 
table.Columns.Add(new DataColumn("state_name", typeof(string)));

// note: if "id_state" is defined as an identity column in your DB,
// row values for that column will be ignored during the bulk copy
table.Rows.Add("1", "Atlanta");
table.Rows.Add("2", "Chicago");
table.Rows.Add("3", "Springfield");

using(SqlBulkCopy bulkCopy = new SqlBulkCopy(connectionString))
{
  bulkCopy.BulkCopyTimeout = 600; // in seconds
  bulkCopy.DestinationTableName = "state";
  bulkCopy.WriteToServer(table);
}
25 голосов
/ 25 марта 2009

Ну, 10 предметов - это не то, что я называю массовым, но для больших наборов SqlBulkCopy - ваш друг. Все, что вам нужно сделать, это указать DataTable или IDataReader (мой предпочтительный вариант, потому что мне нравятся потоковые API). Я сделал нечто подобное здесь (вы можете игнорировать сторону xml - просто создайте подкласс SimpleDataReader).

7 голосов
/ 25 марта 2009

Класс .NET SqlBulkCopy работает довольно хорошо.

2 голосов
/ 25 марта 2009

Я создаю список в виде строки xml и передаю его в сохраненный процесс. В SQL 2005 он расширил функции xml для анализа xml и выполнения массовой вставки.

проверьте это сообщение: Передача списков в SQL Server 2005 с параметрами XML

2 голосов
/ 25 марта 2009

Вот хороший пример SqlBulkCopy в действии:

http://blogs.msdn.com/nikhilsi/archive/2008/06/11/bulk-insert-into-sql-from-c-app.aspx

2 голосов
/ 25 марта 2009

Вы можете создать BLOB (изображение) и отправить его в качестве параметра в хранимую процедуру. Внутри хранимой процедуры вы можете получить все элементы, используя substring ().

1 голос
/ 20 октября 2016

В качестве решения для SqlBulkCopy я создал класс, который принимает Datatable или List<T> и размер буфера (CommitBatchSize). Он преобразует список в таблицу данных, используя расширение (во втором классе).

Работает очень быстро. На моем компьютере я могу вставить более 10 миллионов сложных записей менее чем за 10 секунд.

Вот класс:

using System;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Data.SqlClient;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace DAL
{

public class BulkUploadToSql<T>
{
    public IList<T> InternalStore { get; set; }
    public string TableName { get; set; }
    public int CommitBatchSize { get; set; }=1000;
    public string ConnectionString { get; set; }

    public void Commit()
    {
        if (InternalStore.Count>0)
        {
            DataTable dt;
            int numberOfPages = (InternalStore.Count / CommitBatchSize)  + (InternalStore.Count % CommitBatchSize == 0 ? 0 : 1);
            for (int pageIndex = 0; pageIndex < numberOfPages; pageIndex++)
                {
                    dt= InternalStore.Skip(pageIndex * CommitBatchSize).Take(CommitBatchSize).ToDataTable();
                BulkInsert(dt);
                }
        } 
    }

    public void BulkInsert(DataTable dt)
    {
        using (SqlConnection connection = new SqlConnection(ConnectionString))
        {
            // make sure to enable triggers
            // more on triggers in next post
            SqlBulkCopy bulkCopy =
                new SqlBulkCopy
                (
                connection,
                SqlBulkCopyOptions.TableLock |
                SqlBulkCopyOptions.FireTriggers |
                SqlBulkCopyOptions.UseInternalTransaction,
                null
                );

            // set the destination table name
            bulkCopy.DestinationTableName = TableName;
            connection.Open();

            // write the data in the "dataTable"
            bulkCopy.WriteToServer(dt);
            connection.Close();
        }
        // reset
        //this.dataTable.Clear();
    }

}

public static class BulkUploadToSqlHelper
{
    public static DataTable ToDataTable<T>(this IEnumerable<T> data)
    {
        PropertyDescriptorCollection properties =
            TypeDescriptor.GetProperties(typeof(T));
        DataTable table = new DataTable();
        foreach (PropertyDescriptor prop in properties)
            table.Columns.Add(prop.Name, Nullable.GetUnderlyingType(prop.PropertyType) ?? prop.PropertyType);
        foreach (T item in data)
        {
            DataRow row = table.NewRow();
            foreach (PropertyDescriptor prop in properties)
                row[prop.Name] = prop.GetValue(item) ?? DBNull.Value;
            table.Rows.Add(row);
        }
        return table;
    }
}

}

Вот пример, когда я хочу вставить список моего пользовательского объекта List<PuckDetection> (ListDetections):

var objBulk = new BulkUploadToSql<PuckDetection>()
{
        InternalStore = ListDetections,
        TableName= "PuckDetections",
        CommitBatchSize=1000,
        ConnectionString="ENTER YOU CONNECTION STRING"
};
objBulk.Commit();
1 голос
/ 25 марта 2009

Создайте XML-документ, который содержит все элементы для вставки. Затем внутри хранимой процедуры используйте поддержку TSQL xml ( OPENXML ), чтобы прочитать все данные из документа XML и вставить их в свои таблицы, надеясь, что для каждой таблицы будет использоваться один оператор вставки.

Однако, если вы вставляете данные только в одну таблицу и вам не нужна логика на стороне базы данных, почему бы не использовать SqlBulkCopy ?

1 голос
/ 25 марта 2009

Вы можете обновить документ Xml, Sql 2005 отлично с ними работает. Один узел на строку, но только один параметр для Xml.

1 голос
/ 25 марта 2009

Сохраните ваши данные в текстовый файл с разделителями каналов (или что-то еще, если в ваших данных есть каналы) и используйте Bulk Insert .

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...