12

这几天我一直在努力提高 NHibernate Insert 的性能。

我在很多帖子(例如这篇文章)中读到无状态会话每秒可以插入 1000~2000 条记录......但是它可以插入 1243 条记录的最佳时间对我来说超过 9 秒:

var sessionFactory = new NHibernateConfiguration().CreateSessionFactory();
using (IStatelessSession statelessSession = sessionFactory.OpenStatelessSession())
{
   statelessSession.SetBatchSize(adjustmentValues.Count);

   foreach (var adj in adjustmentValues)
      statelessSession.Insert(adj);
}

班上 :

public partial class AdjustmentValue : PersistentObject, IFinancialValue
{
    public virtual double Amount { get; set; }
    public virtual bool HasManualValue { get; set; }
    public virtual bool HasScaleValue { get; set; }
    public virtual string Formula { get; set; }
    public virtual DateTime IssueDate { get; set; }

    public virtual CompanyTopic CompanyTopic { get; set; }
}

班级地图:

public class AdjustmentValueMap : ClassMap<AdjustmentValue>
{
    public AdjustmentValueMap()
    {
       Id(P => P.Id);

       Map(p => p.Amount);
       Map(p => p.IssueDate);
       Map(p => p.HasManualValue);
       Map(p => p.HasScaleValue);
       Map(p => p.Formula);

       References(p => p.CompanyTopic)
           .Fetch.Join();
    }
}

我错过了什么吗?知道如何加快插入速度吗?

在此处输入图像描述

生成的查询将如下所示:

在此处输入图像描述

4

2 回答 2

20

从您使用身份作为 POID 的 NProf 结果来看。因此,您不能利用批量写入。每个插入/更新/删除都是一个单独的命令。这就是为什么它需要这么长时间。

如果您将 POID 更改为 hilo、guid 或 guid.comb 并将批量大小设置为 500 或 1000,那么您将看到写入时间的显着改善。

于 2011-12-19T15:57:50.060 回答
5

我假设您使用的是 SQL Server 2008。

想到的东西很少。您是否使用身份键(在示例输出中选择 SCOPE_IDENTITY())作为实体的主键?如果是,那么我相信 NHibernate 必须在对象实际保存到数据库之前为每个对象执行 SCOPE_IDENTITY() 调用。因此,如果您要插入 1000 个对象,Nhibernate 将生成 1000 个 INSERT 语句和 1000 个 select SCOPE_IDENTITY() 语句。

我不是 100% 确定,但它也可能会破坏批处理。既然您使用的是 NHProf,那么它说明了什么?它是否表明所有语句都是批处理的,或者您可以在 NProf UI 中选择单个 INSERT 语句?如果您的插入未批处理,那么您很可能会在 NHProf 中看到“大量单独写入”警报。

编辑:

如果您无法更改身份生成,则可以使用SqlBulkCopy。我已经在数据迁移中将它与 NHibernate 一起使用,并且可以正常工作。Ayende Rahien 在他的博客上有示例,可以帮助您入门。

于 2011-12-19T15:43:59.717 回答