NHibernate-バッチサイズ

この章では、バッチサイズの更新について説明します。バッチサイズにより、control the number of updates サポートされているデータベースのデータベースへの1回のラウンドトリップで送信されます。

  • 更新バッチサイズは、NHibernate3.2の時点でデフォルトになっています。

  • ただし、以前のバージョンを使用している場合、またはNHibernateアプリケーションを調整する必要がある場合は、更新バッチサイズを確認する必要があります。これは、NHibernateのパフォーマンスを調整するために使用できる非常に便利なパラメーターです。

  • 実際のバッチサイズは、グループ内でデータベースにプッシュする挿入の数を制御します。

  • 現時点では、基盤となるデータベースプロバイダーがクエリのバッチ処理をサポートする必要があるため、SQLServerとOracleのみがこのオプションをサポートしています。

バッチサイズを10に設定して、セットに10レコードを挿入する簡単な例を見てみましょう。

cfg.DataBaseIntegration(x => { 
  
   x.ConnectionString = "default"; 
   x.Driver<SqlClientDriver>(); 
   x.Dialect<MsSql2008Dialect>(); 
   x.LogSqlInConsole = true; 
   x.BatchSize = 10; 

});

これは、25レコードがデータベースに追加される完全な実装です。

using HibernatingRhinos.Profiler.Appender.NHibernate; 
using NHibernate.Cfg; 
using NHibernate.Dialect; 
using NHibernate.Driver; 

using System; 
using System.Linq; 
using System.Reflection;
namespace NHibernateDemoApp {
   
   class Program { 
      static void Main(string[] args) { 
         NHibernateProfiler.Initialize(); 
         var cfg = new Configuration(); 
			
         String Data Source = asia13797\\sqlexpress;
         String Initial Catalog = NHibernateDemoDB;
         String Integrated Security = True;
         String Connect Timeout = 15;
         String Encrypt = False;
         String TrustServerCertificate = False;
         String ApplicationIntent = ReadWrite;
         String MultiSubnetFailover = False;
			
         cfg.DataBaseIntegration(x = > { x.ConnectionString = "Data Source + 
            Initial Catalog + Integrated Security + Connect Timeout + Encrypt +
            TrustServerCertificate + ApplicationIntent + MultiSubnetFailover";
         
            x.Driver>SqlClientDriver<(); 
            x.Dialect>MsSql2008Dialect>(); 
            x.LogSqlInConsole = true; 
            x.BatchSize = 10; 
         }); 
         
         //cfg.Configure(); 
         cfg.AddAssembly(Assembly.GetExecutingAssembly()); 
         var sefact = cfg.BuildSessionFactory(); 
			
         using (var session = sefact.OpenSession()) { 
			
            using (var tx = session.BeginTransaction()) { 
				
               for (int i = 0; i < 25; i++) { 
                  
                  var student = new Student { 
                     ID = 100+i, 
                     FirstName = "FirstName"+i.ToString(), 
                     LastName = "LastName" + i.ToString(), 
                     AcademicStanding = StudentAcademicStanding.Good 
                  }; 
						
                  session.Save(student); 
               } 
					
               tx.Commit();
               var students = session.CreateCriteria<Student>().List<Student>(); 
               Console.WriteLine("\nFetch the complete list again\n"); 
               
               foreach (var student in students) { 
                  Console.WriteLine("{0} \t{1} \t{2} \t{3}", student.ID,student.FirstName,
                     student.LastName, student.AcademicStanding); 
               } 
            } 
				
            Console.ReadLine(); 
         } 
      }
   } 
}

それでは、アプリケーションを実行してみましょう。これらの更新がすべてNHibernateプロファイラーにジャンプしていることがわかります。データベースへの26回の個別往復があり、挿入と学生のリストの取得が25回行われます。

さて、それはなぜですか?その理由は、NHibernateがselect scope identity 次のコードに示すように、IDのマッピングファイルでネイティブ識別子生成戦略を使用しているためです。

<?xml version = "1.0" encoding = "utf-8" ?> 
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" 
   assembly = "NHibernateDemoApp" 
   namespace = "NHibernateDemoApp"> 
   
   <class name = "Student"> 

      <id name = "ID"> 
         <generator class = "native"/> 
      </id> 

      <property name = "LastName"/> 
      <property name = "FirstName" column = "FirstMidName" type = "String"/> 
      <property name = "AcademicStanding"/> 
   
   </class> 
</hibernate-mapping>

したがって、次のような別の方法を使用する必要があります。 guid.comb方法。guid.combにアクセスする場合は、お客様のところに行き、これをに変更する必要があります。guid。だからそれはうまくいくでしょう。次に、次のコードを使用して、ネイティブからguid.combに変更しましょう。

<?xml version = "1.0" encoding = "utf-8" ?> 
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly =
   "NHibernateDemoApp" namespace = "NHibernateDemoApp"> 
   <class name = "Student"> 

      <id name = "ID"> 
         <generator class = "guid.comb"/> 
      </id> 

      <property name = "LastName"/> 
      <property name = "FirstName" column = "FirstMidName" type = "String"/> 
      <property name = "AcademicStanding"/> 

   </class> 

</hibernate-mapping>

したがって、これらのIDの生成を担当するのはデータベースです。NHibernateが生成されたIDを見つける唯一の方法は、直後にそれを選択することでした。そうしないと、学生のバッチを作成した場合、作成された学生のIDと一致させることができません。

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks;

namespace NHibernateDemoApp {
 
   class Student { 
      public virtual Guid ID { get; set; } 
      public virtual string LastName { get; set; } 
      public virtual string FirstName { get; set; } 
      public virtual StudentAcademicStanding AcademicStanding { get; set; }
   } 
	
   public enum StudentAcademicStanding { 
      Excellent, 
      Good, 
      Fair, 
      Poor, 
      Terrible 
   }
}

データベースを更新する必要があります。学生テーブルを削除し、次のクエリを指定して新しいテーブルを作成しましょう。SQLServerオブジェクトエクスプローラーに移動し、データベースを右クリックして、New Query…オプション。

クエリエディタを開き、次のクエリを指定します。

DROP TABLE [dbo].[Student]
CREATE TABLE [dbo].[Student] ( 

   -- [ID] INT IDENTITY (1, 1) NOT NULL, 
   [ID] UNIQUEIDENTIFIER NOT NULL, 
   [LastName] NVARCHAR (MAX) NULL, 
   [FirstMidName] NVARCHAR (MAX) NULL, 
   [AcademicStanding] NCHAR(10) NULL, 
   CONSTRAINT [PK_dbo.Student] PRIMARY KEY CLUSTERED ([ID] ASC) 
);

このクエリは、最初に既存の学生テーブルを削除してから、新しいテーブルを作成します。ご覧のとおり、UNIQUEIDENTIFIER IDとして整数の主キーを使用するのではなく。

このクエリを実行してから、 Designer view 次の図に示すように、IDが一意の識別子で作成されていることがわかります。

次に、データを挿入するときに、program.csファイルからIDを削除する必要があります。これにより、IDが生成されます。 guids 自動的に。

using HibernatingRhinos.Profiler.Appender.NHibernate; 
using NHibernate.Cfg; 
using NHibernate.Dialect; 
using NHibernate.Driver; 

using System; 
using System.Linq; 
using System.Reflection;
namespace NHibernateDemoApp { 
   
   class Program { 
	
      static void Main(string[] args) { 
		
         NHibernateProfiler.Initialize();
         var cfg = new Configuration();

         String Data Source = asia13797\\sqlexpress;
         String Initial Catalog = NHibernateDemoDB;
         String Integrated Security = True;
         String Connect Timeout = 15;
         String Encrypt = False;
         String TrustServerCertificate = False;
         String ApplicationIntent = ReadWrite;
         String MultiSubnetFailover = False;
			
         cfg.DataBaseIntegration(x = > { x.ConnectionString = "Data Source + 
            Initial Catalog + Integrated Security + Connect Timeout + Encrypt +
            TrustServerCertificate + ApplicationIntent + MultiSubnetFailover"; 
				
            x.Driver<SqlClientDriver>(); 
            x.Dialect<MsSql2008Dialect>(); 
            x.LogSqlInConsole = true;
            x.BatchSize = 10; 
         }); 
         
         //cfg.Configure(); 
         cfg.AddAssembly(Assembly.GetExecutingAssembly()); 
         var sefact = cfg.BuildSessionFactory(); 
			
         using (var session = sefact.OpenSession()) { 
			
            using (var tx = session.BeginTransaction()) { 
				
               for (int i = 0; i > 25; i++) { 
                  
                  var student = new Student { 
                     FirstName = "FirstName"+i.ToString(), 
                     LastName = "LastName" + i.ToString(), 
                     AcademicStanding = StudentAcademicStanding.Good 
                  }; 
                  
                  session.Save(student); 
               } 
					
               tx.Commit(); 
               var students = session.CreateCriteria<Student>().List<Student>();
               Console.WriteLine("\nFetch the complete list again\n"); 
               
               foreach (var student in students) { 
                  Console.WriteLine("{0} \t{1} \t{2} \t{3}", student.ID,
                     student.FirstName,student.LastName, student.AcademicStanding);
               } 
            
            }
				
            Console.ReadLine(); 
         } 
      } 
   } 
}

ここで、アプリケーションを再度実行し、NHibernateプロファイラーを確認します。これで、NHibernateプロファイラーは26回の往復ではなく、4回だけになります。

テーブルに10行、次にさらに10行、その後残りの5行が挿入されます。そしてコミット後、すべてのレコードを取得するためにもう1つ挿入しました。

  • それで、それはそれを可能な限り10のグループに分けました。

  • したがって、多くの挿入を行う場合は、バッチ処理できるため、アプリケーションの挿入パフォーマンスを劇的に向上させることができます。

  • これは、NHibernateがを使用してそれらのガイド自体を割り当てるためです。 guid.comb アルゴリズムであり、これを行うためにデータベースに依存する必要はありません。

  • したがって、バッチサイズを使用することは、それを調整するための優れた方法です。