2015-03-19 SQL Server 批量插入数据的两种方法

本文对比了三种SQLServer批量插入数据的方法:传统Insert语句、Bulk插入和表值参数(TVPs)。通过实例测试,展示了不同方法在插入大量数据时的性能差异。

此文原创自优快云 TJVictor专栏:http://blog.youkuaiyun.com/tjvictor/archive/2009/07/18/4360030.aspx 


  在SQL Server 中插入一条数据使用Insert语句,但是如果想要批量插入一堆数据的话,循环使用Insert不仅效率低,而且会导致SQL一系统性能问题。下面介绍SQL Server支持的两种批量数据插入方法:Bulk和表值参数(Table-Valued Parameters)。

运行下面的脚本,建立测试数据库和表值参数。

[c-sharp] view plaincopy
  1. --Create DataBase  
  2. create database BulkTestDB;  
  3. go  
  4. use BulkTestDB;  
  5. go  
  6. --Create Table  
  7. Create table BulkTestTable(  
  8. Id int primary key,  
  9. UserName nvarchar(32),  
  10. Pwd varchar(16))  
  11. go  
  12. --Create Table Valued  
  13. CREATE TYPE BulkUdt AS TABLE  
  14.   (Id int,  
  15.    UserName nvarchar(32),  
  16.    Pwd varchar(16))  

 

下面我们使用最简单的Insert语句来插入100万条数据,代码如下:

[c-sharp] view plaincopy
  1. Stopwatch sw = new Stopwatch();  
  2.   
  3. SqlConnection sqlConn = new SqlConnection(  
  4.     ConfigurationManager.ConnectionStrings["ConnStr"].ConnectionString);//连接数据库  
  5.   
  6. SqlCommand sqlComm = new SqlCommand();  
  7. sqlComm.CommandText = string.Format("insert into BulkTestTable(Id,UserName,Pwd)values(@p0,@p1,@p2)");//参数化SQL  
  8. sqlComm.Parameters.Add("@p0", SqlDbType.Int);  
  9. sqlComm.Parameters.Add("@p1", SqlDbType.NVarChar);  
  10. sqlComm.Parameters.Add("@p2", SqlDbType.VarChar);  
  11. sqlComm.CommandType = CommandType.Text;  
  12. sqlComm.Connection = sqlConn;  
  13. sqlConn.Open();  
  14. try  
  15. {  
  16.     //循环插入100万条数据,每次插入10万条,插入10次。  
  17.     for (int multiply = 0; multiply < 10; multiply++)  
  18.     {  
  19.         for (int count = multiply * 100000; count < (multiply + 1) * 100000; count++)  
  20.         {  
  21.   
  22.             sqlComm.Parameters["@p0"].Value = count;  
  23.             sqlComm.Parameters["@p1"].Value = string.Format("User-{0}", count * multiply);  
  24.             sqlComm.Parameters["@p2"].Value = string.Format("Pwd-{0}", count * multiply);  
  25.             sw.Start();  
  26.             sqlComm.ExecuteNonQuery();  
  27.             sw.Stop();  
  28.         }  
  29.         //每插入10万条数据后,显示此次插入所用时间  
  30.         Console.WriteLine(string.Format("Elapsed Time is {0} Milliseconds", sw.ElapsedMilliseconds));  
  31.     }  
  32. }  
  33. catch (Exception ex)  
  34. {  
  35.     throw ex;  
  36. }  
  37. finally  
  38. {  
  39.     sqlConn.Close();  
  40. }  
  41.   
  42. Console.ReadLine();  

耗时图如下:

使用Insert语句插入10万数据的耗时图

由于运行过慢,才插入10万条就耗时72390 milliseconds,所以我就手动强行停止了。

 

下面看一下使用Bulk插入的情况:

bulk方法主要思想是通过在客户端把数据都缓存在Table中,然后利用SqlBulkCopy一次性把Table中的数据插入到数据库

代码如下:

[c-sharp] view plaincopy
  1. public static void BulkToDB(DataTable dt)  
  2. {  
  3.     SqlConnection sqlConn = new SqlConnection(  
  4.         ConfigurationManager.ConnectionStrings["ConnStr"].ConnectionString);  
  5.     SqlBulkCopy bulkCopy = new SqlBulkCopy(sqlConn);  
  6.     bulkCopy.DestinationTableName = "BulkTestTable";  
  7.     bulkCopy.BatchSize = dt.Rows.Count;  
  8.   
  9.     try  
  10.     {  
  11.         sqlConn.Open();  
  12.     if (dt != null && dt.Rows.Count != 0)  
  13.         bulkCopy.WriteToServer(dt);  
  14.     }  
  15.     catch (Exception ex)  
  16.     {  
  17.         throw ex;  
  18.     }  
  19.     finally  
  20.     {  
  21.         sqlConn.Close();  
  22.         if (bulkCopy != null)  
  23.             bulkCopy.Close();  
  24.     }  
  25. }  
  26.   
  27. public static DataTable GetTableSchema()  
  28. {  
  29.     DataTable dt = new DataTable();  
  30.     dt.Columns.AddRange(new DataColumn[]{  
  31.         new DataColumn("Id",typeof(int)),  
  32.         new DataColumn("UserName",typeof(string)),  
  33.     new DataColumn("Pwd",typeof(string))});  
  34.   
  35.     return dt;  
  36. }  
  37.   
  38. static void Main(string[] args)  
  39. {  
  40.     Stopwatch sw = new Stopwatch();  
  41.     for (int multiply = 0; multiply < 10; multiply++)  
  42.     {  
  43.         DataTable dt = Bulk.GetTableSchema();  
  44.         for (int count = multiply * 100000; count < (multiply + 1) * 100000; count++)  
  45.         {  
  46.             DataRow r = dt.NewRow();  
  47.             r[0] = count;  
  48.             r[1] = string.Format("User-{0}", count * multiply);  
  49.             r[2] = string.Format("Pwd-{0}", count * multiply);  
  50.             dt.Rows.Add(r);  
  51.         }  
  52.         sw.Start();  
  53.         Bulk.BulkToDB(dt);  
  54.         sw.Stop();  
  55.         Console.WriteLine(string.Format("Elapsed Time is {0} Milliseconds", sw.ElapsedMilliseconds));  
  56.     }  
  57.   
  58.     Console.ReadLine();  
  59. }  

耗时图如下:

使用Bulk插入100万数据的耗时图

可见,使用Bulk后,效率和性能明显上升。使用Insert插入10万数据耗时72390,而现在使用Bulk插入100万数据才耗时17583。

 

最后再看看使用表值参数的效率,会另你大为惊讶的。

 

表值参数是SQL Server 2008新特性,简称TVPs。对于表值参数不熟悉的朋友,可以参考最新的book online,我也会另外写一篇关于表值参数的博客,不过此次不对表值参数的概念做过多的介绍。言归正传,看代码:

[c-sharp] view plaincopy
  1. public static void TableValuedToDB(DataTable dt)  
  2. {  
  3.     SqlConnection sqlConn = new SqlConnection(  
  4.       ConfigurationManager.ConnectionStrings["ConnStr"].ConnectionString);  
  5.     const string TSqlStatement =  
  6.      "insert into BulkTestTable (Id,UserName,Pwd)" +  
  7.      " SELECT nc.Id, nc.UserName,nc.Pwd" +  
  8.      " FROM @NewBulkTestTvp AS nc";  
  9.     SqlCommand cmd = new SqlCommand(TSqlStatement, sqlConn);  
  10.     SqlParameter catParam = cmd.Parameters.AddWithValue("@NewBulkTestTvp", dt);  
  11.     catParam.SqlDbType = SqlDbType.Structured;  
  12.     //表值参数的名字叫BulkUdt,在上面的建立测试环境的SQL中有。  
  13.     catParam.TypeName = "dbo.BulkUdt";  
  14.     try  
  15.     {  
  16.       sqlConn.Open();  
  17.       if (dt != null && dt.Rows.Count != 0)  
  18.       {  
  19.           cmd.ExecuteNonQuery();  
  20.       }  
  21.     }  
  22.     catch (Exception ex)  
  23.     {  
  24.       throw ex;  
  25.     }  
  26.     finally  
  27.     {  
  28.       sqlConn.Close();  
  29.     }  
  30. }  
  31.   
  32. public static DataTable GetTableSchema()  
  33. {  
  34.     DataTable dt = new DataTable();  
  35.     dt.Columns.AddRange(new DataColumn[]{  
  36.       new DataColumn("Id",typeof(int)),  
  37.       new DataColumn("UserName",typeof(string)),  
  38.       new DataColumn("Pwd",typeof(string))});  
  39.   
  40.     return dt;  
  41. }  
  42.   
  43. static void Main(string[] args)  
  44. {  
  45.     Stopwatch sw = new Stopwatch();  
  46.     for (int multiply = 0; multiply < 10; multiply++)  
  47.     {  
  48.         DataTable dt = TableValued.GetTableSchema();  
  49.         for (int count = multiply * 100000; count < (multiply + 1) * 100000; count++)  
  50.         {          
  51.             DataRow r = dt.NewRow();  
  52.             r[0] = count;  
  53.             r[1] = string.Format("User-{0}", count * multiply);  
  54.             r[2] = string.Format("Pwd-{0}", count * multiply);  
  55.             dt.Rows.Add(r);  
  56.         }  
  57.         sw.Start();  
  58.         TableValued.TableValuedToDB(dt);  
  59.         sw.Stop();  
  60.         Console.WriteLine(string.Format("Elapsed Time is {0} Milliseconds", sw.ElapsedMilliseconds));  
  61.     }  
  62.   
  63.     Console.ReadLine();  
  64. }  

耗时图如下:

使用表值参数插入100万数据的耗时图

比Bulk还快5秒。

<think>好的,我现在要帮用户介绍数据库的SQL插入语句。首先,用户之前问过关于C++计算子串次数的问题,现在转向SQL插入,可能是在学习编程或数据库相关知识,需要基础的操作指导。 用户的问题是关于SQL插入语句的,我需要先确定用户的需求层次。他们可能是刚开始学习数据库,需要基本的语法和示例,或者可能有更深入的需求,比如批量插入、处理冲突等。由于用户的问题比较基础,我应该先覆盖最常用的INSERT INTO用法,然后逐步扩展。 接下来,我需要考虑常见的插入语句类型。最基本的是插入单行数据,指定列名和值。这部分需要给出清晰的语法结构和示例,比如插入到学生表中。同时,用户可能想知道如果不指定所有列会发生什么,所以需要说明未指定的列会被设置为默认值或NULL,但前提是这些列允许NULL。 然后是省略列名的插入方式,但必须按表结构顺序提供所有列的值。这时候需要注意如果表结构变化,这种方式可能会有问题,所以建议还是指定列名更安全。需要给出例子,并提醒注意事项。 批量插入是另一个重要方面,用户可能需要一次性插入多行数据,提高效率。这里可以用多个VALUES子句,或者结合SELECT语句从其他表导入数据。示例代码需要展示这两种情况,比如插入多个学生记录,或者将符合条件的数据从旧表导入新表。 接下来是处理插入冲突的情况,特别是主键或唯一约束冲突。不同的数据库有不同的处理方法,比如MySQL的ON DUPLICATE KEY UPDATE和PostgreSQL的ON CONFLICT。这里需要分别说明,并给出示例,这样用户知道如何避免插入重复数据时出错。 特殊情况的处理也很重要,比如插入默认值、使用函数或表达式,以及插入大量数据时的优化。用户可能在实际应用中遇到这些情况,需要简要介绍并提供示例,比如使用DEFAULT关键字,或者在插入时调用函数生成数据。 最后,总结关键点,强调基本语法、批量操作、冲突处理和性能优化。还要提醒用户注意SQL注入问题,特别是在动态生成SQL语句时,使用参数化查询来提高安全性。 需要确保整个回答结构清晰,分点说明,每种情况都有示例代码,并且语法正确。同时,使用中文,避免技术术语过于复杂,让初学者容易理解。可能用户是学生或刚入门的开发者,所以需要由浅入深,覆盖常用场景,并提供实用建议。</think>以下是 SQL 插入语句的 **核心用法详解** 和 **常见场景示例**: --- ### 一、基本语法 ```sql INSERT INTO 表名 (列1, 列2, ...) VALUES (值1, 值2, ...); ``` #### 示例 1:插入单行完整数据 ```sql -- 插入学生信息 INSERT INTO Students (id, name, age, major) VALUES (1, '张三', 20, '计算机科学'); ``` #### 示例 2:插入部分列数据 ```sql --插入必填字段(未指定的列需允许 NULL 或存在默认值) INSERT INTO Students (name, major) VALUES ('李四', '数据科学'); ``` --- ### 二、省略列名的插入 ```sql INSERT INTO 表名 VALUES (值1, 值2, ...); -- 必须按表结构顺序提供所有列的值 ``` #### 示例: ```sql INSERT INTO Students VALUES (2, '王五', 22, '人工智能'); ``` > **注意**:表结构变更时此写法易出错,建议显式指定列名 --- ### 三、批量插入 ```sql INSERT INTO 表名 (列1, 列2, ...) VALUES (值1, 值2, ...), (值1, 值2, ...), ...; ``` #### 示例: ```sql INSERT INTO Students (name, age) VALUES ('赵六', 21), ('陈七', 19), ('周八', 23); ``` --- ### 四、从其他表插入数据 ```sql INSERT INTO 目标表 (列1, 列2, ...) SELECT 源列1, 源列2, ... FROM 源表 WHERE 条件; ``` #### 示例:将旧表数据迁移到新表 ```sql INSERT INTO NewStudents (student_name, department) SELECT name, major FROM OldStudents WHERE enrollment_year = 2023; ``` --- ### 五、特殊插入场景 #### 1. 插入默认值 ```sql INSERT INTO Orders (order_id, order_date, status) VALUES (1001, DEFAULT, '待处理'); -- 使用列的默认值 ``` #### 2. 插入时使用函数 ```sql INSERT INTO Logs (event_time, message) VALUES (NOW(), '系统启动'); -- 使用数据库时间函数 ``` --- ### 六、冲突处理(不同数据库语法不同) #### MySQL:`ON DUPLICATE KEY UPDATE` ```sql INSERT INTO Users (id, email) VALUES (1, 'user@example.com') ON DUPLICATE KEY UPDATE email = VALUES(email); ``` #### PostgreSQL:`ON CONFLICT` ```sql INSERT INTO Products (sku, price) VALUES ('X123', 99.99) ON CONFLICT (sku) DO UPDATE SET price = EXCLUDED.price; ``` --- ### 关键总结 1. **基础原则** - 列与值必须数量匹配、类型兼容 - 字符串和日期类型需用单引号包裹 2. **性能优化** - 批量插入比多次单行插入效率高 -数据插入时建议禁用索引和约束检查 3. **安全注意** - 动态拼接 SQL 时需防范 SQL 注入 - 优先使用参数化查询(Prepared Statements) 需要了解具体数据库(如 Oracle/SQL Server)的特殊语法或高级功能,可进一步说明需求!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值