Ускорьте операции массовой вставки с помощью NHibernate - PullRequest
15 голосов
/ 30 марта 2012

Я хочу ускорить массовые insert операции с NHibernate 3.2 в Oracle 11g. Для этого я попытался

Session.Save(entity);
Session.Flush();
Session.Clear();

... в моем цикле foreach, но возникла исключительная ситуация, вызванная отсутствием объектов в сеансе:

не удалось лениво инициализировать коллекцию ролей: MyClass.PropertyX, ни один сеанс или сеанс не был закрыт

Еще одна попытка была установить размер пакета:

<hibernate-configuration xmlns="urn:nhibernate-configuration-2.2">
  <session-factory>
    <property name="connection.provider">NHibernate.Connection.DriverConnectionProvider</property>
    <property name="connection.driver_class">NHibernate.Driver.OracleClientDriver</property>
    <property name="connection.connection_string">xxx</property>
    <property name="dialect">NHibernate.Dialect.Oracle10gDialect</property>
    <property name="adonet.batch_size">50</property>
    <property name="query.substitutions">true=1, false=0</property>
    <property name="proxyfactory.factory_class">NHibernate.Bytecode.DefaultProxyFactoryFactory, NHibernate</property>
  </session-factory>
</hibernate-configuration>

дополнительно я установил Session.SetBatchSize(50) в своем коде и получил следующее исключение:

Для фабрики сеансов не определен размер пакета, пакет отключен. Установите adonet.batch_size = 1, чтобы включить пакетирование.

Единственное место, где выбрасывается это исключение: NonBatchingBatcher , поэтому похоже, что в моем сеансе неправильный дозатор.

Что здесь не так? Как ускорить пакетную вставку с помощью NHibernate (без использования статических сеансов)?

Ответы [ 6 ]

36 голосов
/ 04 апреля 2012

Следующее должно работать,

var testObjects = CreateTestObjects(500000);

var stopwatch = new Stopwatch();
stopwatch.Start();
using (IStatelessSession session = sessionFactory.OpenStatelessSession())
using (ITransaction transaction = session.BeginTransaction())
{
    foreach (var testObject in testObjects)
        session.Insert(testObject);
    transaction.Commit();
}

stopwatch.Stop();
var time = stopwatch.Elapsed;

Ссылка: http://nhibernate.info/blog/2008/10/30/bulk-data-operations-with-nhibernate-s-stateless-sessions.html

5 голосов
/ 12 мая 2014

Все вышеперечисленные советы очень актуальны и очень полезны.Хотел добавить один в коллекцию: отключить ведение журнала .Отображение вашего SQL в консоли заметно замедляет вас, также как и профилирование с использованием NHProf, автоматическое комментирование и красивое форматирование SQL, записанного в журнал через NLog или log4net.В нашем случае:

cfg.AutoCommentSql = false;
cfg.LogFormattedSql = false;

уменьшило время массовой вставки с ~ 6 секунд до чуть более 1 секунды.Поэтому, хотя ведение журнала потенциально может помочь вам решить более серьезные проблемы, оно само по себе идет на снижение производительности!

4 голосов
/ 09 апреля 2012
1 голос
/ 02 апреля 2012

Почему вы очищаете сеанс?

Я думаю, вы не должны очищать сессию в цикле. Чтобы убедиться, что изменения записаны в базу данных, я бы предпочел использовать транзакцию.

псевдокод:

foreach (var i in allElements)
{
    using (var tx = session.BeginTransaction())
    {
        ... do what you have to do with the object
        tx.Commit();
    }
}

Чтобы ускорить процесс, есть и другие вещи, которые могут помочь - вы должны определить, что вы действительно хотите сделать в цикле.

0 голосов
/ 15 июня 2018

Код ниже работает для меня при вставке нескольких составных сущностей

 public static void SqlBulkInsert(this ISession session, DataTable dataTable, string tableName)
    {
        var conn = (SqlConnection)session.Connection;
        using (var cmd = new SqlCommand())
        {
            session.Transaction.Enlist(cmd);
            using (var copy = new SqlBulkCopy(conn, SqlBulkCopyOptions.FireTriggers, cmd.Transaction))
            {
                copy.BulkCopyTimeout = 10000;
                copy.DestinationTableName = tableName;
                foreach (DataColumn column in dataTable.Columns)
                {
                    copy.ColumnMappings.Add(column.ColumnName, column.ColumnName);
                }

                copy.WriteToServer(dataTable);
                copy.Close();
            }
        }
    }

Вам нужно будет создать метод для заполнения объекта DataTable из объекта составной сущности, который вы хотите сохранить.

0 голосов
/ 09 февраля 2017

Я знаю, что вопрос был об Oracle, но для SQL-сервера я собирался написать подпрограмму для получения сопоставлений классов и создания DataTable для использования SqlBulkInsert, но я обнаружил, что кто-то уже сделал это.

https://kaylaniam.wordpress.com/2015/03/13/nhibernate-and-sqlbulkcopy/

Это, вероятно, самый быстрый способ выполнения массовых вставок в SQL Server.

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