Entity Framework與ADO.NET批量插入數據性能測試

Entity Framework是.NET平臺下的一種簡單易用的ORM框架,它既便於Domain Model和持久層的OO設計,也提升了代碼的可維護性。但在使用中發現,有幾類業務場景是EF不太擅長的,好比批量寫入大量同類數據,爲此本人作了一些對比測試,以供你們參考。html

現假設咱們須要作一個用戶批量導入的功能,須要從某處導入1k~1w個User到SQLServer數據庫,本人據說過的常見作法有以下幾種:sql

  1. 使用ADO.NET單條SqlCommand執行1w次(根據常識做爲EF的替代其性能還未入流,因此就不作測試了)
  2. 使用StringBuilder拼接SQL語句,將1w條Insert語句拼接成1到若干條SqlCommand執行
  3. 使用EntityFramework的基本功能進行插入
  4. 使用SqlBulkCopy進行批量插入
  5. 使用存儲過程,其中的2種分支分別對應上述一、2用例,另外還有1種表參數存儲過程。

數據庫準備工做:數據庫

 1 CREATE DATABASE BulkInsertTest
 2 GO
 3 
 4 USE BulkInsertTest
 5 GO
 6 
 7 CREATE TABLE [dbo].[User](
 8     [Id] [int] IDENTITY(1,1) NOT NULL,
 9     [Name] [nvarchar](50) NOT NULL,
10     [Birthday] [date] NOT NULL,
11     [Gender] [char](1) NOT NULL,
12     [Email] [nvarchar](50) NOT NULL,
13     [Deleted] [bit] NOT NULL,
14  CONSTRAINT [PK_User] PRIMARY KEY CLUSTERED 
15 (
16     [Id] ASC
17 )WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]
18 ) ON [PRIMARY]
19 
20 GO
21 
22 CREATE PROCEDURE [dbo].[InsertUser] 
23     @Name nvarchar(50)
24            ,@Birthday date
25            ,@Gender char(1)
26            ,@Email nvarchar(50)
27            ,@Deleted bit
28 AS
29 BEGIN
30     INSERT INTO [BulkInsertTest].[dbo].[User]
31            ([Name]
32            ,[Birthday]
33            ,[Gender]
34            ,[Email]
35            ,[Deleted])
36      VALUES
37            (@Name,@Birthday,@Gender,@Email,@Deleted)
38 
39 END
40 
41 /* Create a table type. */
42 CREATE TYPE LocationTableType AS TABLE 
43 ( Name nvarchar(50)
44            ,Birthday date
45            ,Gender char(1)
46            ,Email nvarchar(50)
47            ,Deleted bit );
48 GO
49 
50 /* Create a procedure to receive data for the table-valued parameter. */
51 CREATE PROCEDURE [dbo].[InsertUsers]
52     @Users LocationTableType
53     AS 
54     SET NOCOUNT ON
55     INSERT INTO [dbo].[User]
56            ([Name]
57            ,[Birthday]
58            ,[Gender]
59            ,[Email]
60            ,[Deleted])
61         SELECT *
62         FROM  @Users;
63 
64 GO
View Code

建立DbContext和User Entity的C#代碼:app

 1 using System;
 2 using System.ComponentModel.DataAnnotations;
 3 using System.ComponentModel.DataAnnotations.Schema;
 4 using System.Data.Entity;
 5 
 6 namespace ConsoleApplication5
 7 {
 8     public class MyDbContext : DbContext
 9     {
10         public MyDbContext() : base("MyDbContext") { }
11 
12         public MyDbContext(string connectionString) :
13             base(connectionString)
14         {
15             
16         }
17 
18         public DbSet<User> Users { get; set; }
19     }
20 
21     [Table("User")]
22     public class User
23     {
24         [Key]
25         public int Id { get; set; }
26 
27         public string Name { get; set; }
28 
29         public DateTime Birthday { get; set; }
30 
31         public string Gender { get; set; }
32 
33         public string Email { get; set; }
34 
35         public bool Deleted { get; set; }
36     }
37 }
View Code

測試程序C#代碼:框架

  1 using System;
  2 using System.Data;
  3 using System.Data.SqlClient;
  4 using System.Diagnostics;
  5 using System.Linq;
  6 using System.Text;
  7 
  8 namespace ConsoleApplication5
  9 {
 10     class Program
 11     {
 12         private const string ConnectionString = "Data Source=.;Initial Catalog=BulkInsertTest;User=sa;Password=IGTtest1";
 13         private const int Times = 10;
 14         private const int Entries = 10000;
 15 
 16         static void Main(string[] args)
 17         {
 18             long sumBulkCopyTime = 0, sumSqlCmdsTime = 0, sumMultiSpTime = 0, sumTableSpTime = 0, sumEfTime = 0;
 19             long maxBulkCopyTime = 0, maxSqlCmdsTime = 0, maxMultiSpTime = 0, maxTableSpTime = 0, maxEfTime = 0;
 20             for (int i = 0; i < Times; i++)
 21             {
 22                 long bulkCopyTime = InsertBySqlBulkCopy();
 23                 sumBulkCopyTime += bulkCopyTime;
 24                 maxBulkCopyTime = Math.Max(maxBulkCopyTime, bulkCopyTime);
 25 
 26                 long sqlCmdsTime = InsertBySqlCmds();
 27                 sumSqlCmdsTime += sqlCmdsTime;
 28                 maxSqlCmdsTime = Math.Max(maxSqlCmdsTime, sqlCmdsTime);
 29 
 30                 long multiSpTime = InsertByMultiStoreProcedure();
 31                 sumMultiSpTime += multiSpTime;
 32                 maxMultiSpTime = Math.Max(maxMultiSpTime, multiSpTime);
 33 
 34                 long tableSpTime = InsertByTableStoreProcedure();
 35                 sumTableSpTime += tableSpTime;
 36                 maxTableSpTime = Math.Max(maxTableSpTime, tableSpTime);
 37 
 38                 long efTime = InsertByEntityFramework();
 39                 sumEfTime += efTime;
 40                 maxEfTime = Math.Max(maxEfTime, efTime);
 41             }
 42             Console.WriteLine(new string('-', 40));
 43             Console.WriteLine("Time Cost of SqlBulkCopy:            avg:{0}ms, max:{1}ms", sumBulkCopyTime / Times, maxBulkCopyTime);
 44             Console.WriteLine("Time Cost of SqlCommands:            avg:{0}ms, max:{1}ms", sumSqlCmdsTime / Times, maxSqlCmdsTime);
 45             Console.WriteLine("Time Cost of MultiStoreProcedure:    avg:{0}ms, max:{1}ms", sumMultiSpTime / Times, maxMultiSpTime);
 46             Console.WriteLine("Time Cost of TableStoreProcedure:    avg:{0}ms, max:{1}ms", sumTableSpTime / Times, maxTableSpTime);
 47             Console.WriteLine("Time Cost of EntityFramework:        avg:{0}ms, max:{1}ms", sumEfTime / Times, maxEfTime);
 48             Console.ReadLine();
 49         }
 50 
 51         private static long InsertBySqlCmds()
 52         {
 53             Stopwatch stopwatch = Stopwatch.StartNew();
 54             using (var connection = new SqlConnection(ConnectionString))
 55             {
 56                 SqlTransaction transaction = null;
 57                 connection.Open();
 58                 try
 59                 {
 60                     transaction = connection.BeginTransaction();
 61                     StringBuilder sb = new StringBuilder();
 62                     for (int j = 0; j < Entries; j++)
 63                     {
 64                         sb.AppendFormat(@"INSERT INTO dbo.[User] ([Name],[Birthday],[Gender],[Email],[Deleted])
 65 VALUES('{0}','{1:yyyy-MM-dd}','{2}','{3}',{4});", "name" + j, DateTime.Now.AddDays(j), 'M', "user" + j + "@abc.com", 0);
 66                     }
 67                     var sqlCmd = connection.CreateCommand();
 68                     sqlCmd.CommandText = sb.ToString();
 69                     sqlCmd.Transaction = transaction;
 70                     sqlCmd.ExecuteNonQuery();
 71                     transaction.Commit();
 72                 }
 73                 catch
 74                 {
 75                     if (transaction != null)
 76                     {
 77                         transaction.Rollback();
 78                     }
 79                     throw;
 80                 }
 81             }
 82             stopwatch.Stop();
 83             Console.WriteLine("SqlCommand time cost: {0}ms", stopwatch.ElapsedMilliseconds);
 84             return stopwatch.ElapsedMilliseconds;
 85         }
 86 
 87         private static long InsertByMultiStoreProcedure()
 88         {
 89             Stopwatch stopwatch = Stopwatch.StartNew();
 90             using (var connection = new SqlConnection(ConnectionString))
 91             {
 92                 SqlTransaction transaction = null;
 93                 connection.Open();
 94                 for (int i = 0; i < 10; i++)
 95                 {
 96                     try
 97                     {
 98                         transaction = connection.BeginTransaction();
 99                         StringBuilder sb = new StringBuilder();
100                         for (int j = 0; j < Entries/10; j++)
101                         {
102                             sb.AppendFormat(@"EXECUTE [dbo].[InsertUser] '{0}','{1:yyyy-MM-dd}','{2}','{3}',{4};",
103                                             "name" + j, DateTime.Now.AddDays(j), 'M', "user" + j + "@abc.com", 0);
104                         }
105                         var sqlCmd = connection.CreateCommand();
106                         sqlCmd.CommandText = sb.ToString();
107                         sqlCmd.Transaction = transaction;
108                         sqlCmd.ExecuteNonQuery();
109                         transaction.Commit();
110                     }
111                     catch
112                     {
113                         if (transaction != null)
114                         {
115                             transaction.Rollback();
116                         }
117                         throw;
118                     }
119                 }
120             }
121             stopwatch.Stop();
122             Console.WriteLine("MultiStoreProcedure time cost: {0}ms", stopwatch.ElapsedMilliseconds);
123             return stopwatch.ElapsedMilliseconds;
124         }
125 
126         private static long InsertByTableStoreProcedure()
127         {
128             Stopwatch stopwatch = Stopwatch.StartNew();
129             var table = PrepareDataTable();
130             using (var connection = new SqlConnection(ConnectionString))
131             {
132                 SqlTransaction transaction = null;
133                 connection.Open();
134                 try
135                 {
136                     transaction = connection.BeginTransaction();
137                     var sqlCmd = connection.CreateCommand();
138                     sqlCmd.CommandText = "InsertUsers";
139                     sqlCmd.CommandType = CommandType.StoredProcedure;
140                     sqlCmd.Parameters.Add(new SqlParameter("@Users", SqlDbType.Structured));
141                     sqlCmd.Parameters["@Users"].Value = table;
142                     sqlCmd.Transaction = transaction;
143                     sqlCmd.ExecuteNonQuery();
144                     transaction.Commit();
145                 }
146                 catch
147                 {
148                     if (transaction != null)
149                     {
150                         transaction.Rollback();
151                     }
152                     throw;
153                 }
154             }
155             stopwatch.Stop();
156             Console.WriteLine("TableStoreProcedure time cost: {0}ms", stopwatch.ElapsedMilliseconds);
157             return stopwatch.ElapsedMilliseconds;
158         }
159 
160         private static long InsertBySqlBulkCopy()
161         {
162             Stopwatch stopwatch = Stopwatch.StartNew();
163 
164             var table = PrepareDataTable();
165             SqlBulkCopy(table);
166 
167             stopwatch.Stop();
168             Console.WriteLine("SqlBulkCopy time cost: {0}ms", stopwatch.ElapsedMilliseconds);
169             return stopwatch.ElapsedMilliseconds;
170         }
171 
172         private static DataTable PrepareDataTable()
173         {
174             DataTable table = new DataTable();
175             table.Columns.Add("Name", typeof (string));
176             table.Columns.Add("Birthday", typeof (DateTime));
177             table.Columns.Add("Gender", typeof (char));
178             table.Columns.Add("Email", typeof (string));
179             table.Columns.Add("Deleted", typeof (bool));
180             for (int i = 0; i < Entries; i++)
181             {
182                 var row = table.NewRow();
183                 row["Name"] = "name" + i;
184                 row["Birthday"] = DateTime.Now.AddDays(i);
185                 row["Gender"] = 'M';
186                 row["Email"] = "user" + i + "@abc.com";
187                 row["Deleted"] = false;
188                 table.Rows.Add(row);
189             }
190             return table;
191         }
192 
193         private static void SqlBulkCopy(DataTable dataTable)
194         {
195             using (var connection = new SqlConnection(ConnectionString))
196             {
197                 SqlTransaction transaction = null;
198                 connection.Open();
199                 try
200                 {
201                     transaction = connection.BeginTransaction();
202                     using (var sqlBulkCopy = new SqlBulkCopy(connection, SqlBulkCopyOptions.Default, transaction))
203                     {
204                         sqlBulkCopy.BatchSize = dataTable.Rows.Count;
205 
206                         sqlBulkCopy.DestinationTableName = "[User]";
207                         //sqlBulkCopy.ColumnMappings.Add("Id", "Id");
208                         sqlBulkCopy.ColumnMappings.Add("Name", "Name");
209                         sqlBulkCopy.ColumnMappings.Add("Birthday", "Birthday");
210                         sqlBulkCopy.ColumnMappings.Add("Gender", "Gender");
211                         sqlBulkCopy.ColumnMappings.Add("Email", "Email");
212                         sqlBulkCopy.ColumnMappings.Add("Deleted", "Deleted");
213                         
214                         sqlBulkCopy.WriteToServer(dataTable);
215                     }
216                     transaction.Commit();
217                 }
218                 catch
219                 {
220                     if (transaction!=null)
221                     {
222                         transaction.Rollback();
223                     }
224                     throw;
225                 }
226             }
227         }
228 
229         private static long InsertByEntityFramework()
230         {
231             Stopwatch stopwatch = Stopwatch.StartNew();
232             using (MyDbContext context = new MyDbContext(ConnectionString))
233             {
234                 context.Configuration.AutoDetectChangesEnabled = false;
235                 context.Configuration.ValidateOnSaveEnabled = false;
236                 for (int i = 0; i < Entries; i++)
237                 {
238                     context.Users.Add(new User()
239                                            {
240                                                Name = "name" + i,
241                                                Birthday = DateTime.Now.AddDays(i),
242                                                Gender = "F",
243                                                Email = "user" + i + "@abc.com",
244                                                Deleted = false
245                                            });
246                 }
247                 context.SaveChanges();
248             }
249 
250             stopwatch.Stop();
251             Console.WriteLine("EntityFramework time cost: {0}ms", stopwatch.ElapsedMilliseconds);
252             return stopwatch.ElapsedMilliseconds;
253         }
254     }
255 }
View Code

插入1000行測試結果:ide

插入10000行測試結果:性能

分析與結論:單從性能上來講,SqlBulkCopy和表參數StoreProcedure勝出,且完勝Entity Framework,因此當EF實在沒法知足性能要求時,SqlBulkCopy或表參數SP能夠很好的解決EF批量插入的性能問題。但衡量軟件產品的標準不單單隻有性能這一方面,好比咱們還要在設計美學和性能之間進行權衡。當插入數據量較小或是低壓力時間段自動執行插入的話,EF仍然是不錯的選擇。從代碼可維護性方面來看ADO.NET實現的可讀性、重構友好型都弱於EF實現,因此對於需求變更較多的領域模型而言這幾種解決方法都須要更多的設計抽象和單元測試,以此來確保產品的持續發展。從影響範圍來看,在ADO.NET實現方式中SqlBulkCopy和拼接Sql字符串的方案不須要額外加入存儲過程,因此能夠在不影響數據庫部署的前提下與EF的實現相互替換。單元測試

 

關於SqlBulkCopy請參考:Bulk Copy Operations in SQL Server測試

爲了比較優雅使用SqlBulkCopy,有人寫了一種AsDataReader擴展方法請參考:LinqEntityDataReaderui

根據MSDN的說法,因爲表參數存儲過程的啓動準備消耗時間較小,因此1k行(經驗)如下插入性能將勝於SqlBulkCopy,而隨着插入行數的增多,SqlBulkCopy的性能優點將體現出來,另外兩種方案相比還有一些其餘方面的差別,從本測試的實際結果來看,SqlBulkCopy在首次插入1k條數據時確實耗時稍長一點。具體請參考:Table-Valued Parameters vs. BULK INSERT Operations

另外還有人作過SqlBulkCopy和SqlDataAdapter插入的性能對比:High performance bulk loading to SQL Server using SqlBulkCopy

相關文章
相關標籤/搜索