内容显示页
 
类别:.Net + C# | 浏览(3812) | 2010-3-14 14:34:30

刚进公司时,领导说物流水系统有一个问题:发送水票速度很慢,每发送一次就得等10多分钟,问我有没有解决方法,时间原因一直没去研究。

今天早上reader 上收到cnblogs的订阅里看到一个关于SQL语句快速插入的文章,提到SqlBulkCopy,感觉不错,按他的测试SqlBulkCopy要比普通插入快近30倍,

按这个来算,我们那个发水票的时间就会由 10分钟-->20秒,这可太神奇了。

于是乎,下demo,测试,改成自己一般使用的方法测试,NND,还真可以说是极速。

 在此贴上我的Demo:UploadFiles//SqlBulkCopy.rar

using System;
using System.Diagnostics;
using System.Data;
using System.Data.SqlClient;
using Microsoft.ApplicationBlocks.Data;
using System.Text;

namespace ConsoleAppInsertTest
{
    class Program
    {
        static int count = 100000;           //插入的条数
        static void Main(string[] args)
        {
            long runTime = 0;

            SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "TRUNCATE TABLE dbo.Passport");
            runTime = SqlBulkCopyInsert();
            Console.WriteLine(string.Format("使用SqlBulkCopy插入{1}条数据所用的时间是{0}毫秒", runTime, count));

            SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "TRUNCATE TABLE dbo.Passport");
            runTime = TransactionInsert2();
            Console.WriteLine(string.Format("       使用事务插入{1}条数据所用的时间是{0}毫秒  --使用事务插入数据,分组执行", runTime, count));

            SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "TRUNCATE TABLE dbo.Passport");
            runTime = TransactionInsert1();
            Console.WriteLine(string.Format("       使用事务插入{1}条数据所用的时间是{0}毫秒  --先存到string再一次性执行", runTime, count));

            SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "TRUNCATE TABLE dbo.Passport");
            runTime = TransactionInsert();
            Console.WriteLine(string.Format("       使用事务插入{1}条数据所用的时间是{0}毫秒  --在事务里一条一条执行", runTime, count));


            SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "TRUNCATE TABLE dbo.Passport");
            runTime = CommonInsert1();
            Console.WriteLine(string.Format("       普通方式插入{1}条数据所用的时间是{0}毫秒 --保持SqlConnection", runTime, count));

            SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "TRUNCATE TABLE dbo.Passport");
            runTime = CommonInsert();
            Console.WriteLine(string.Format("       普通方式插入{1}条数据所用的时间是{0}毫秒  --一条一条插入", runTime, count));

            //使用SqlBulkCopy插入100000条数据所用的时间是1301毫秒
            //       使用事务插入100000条数据所用的时间是5811毫秒  --使用事务插入数据,分组执行
            //       使用事务插入100000条数据所用的时间是23315毫秒  --先存到string再一次性执行
            //       使用事务插入100000条数据所用的时间是15756毫秒  --在事务里一条一条执行
            //       普通方式插入100000条数据所用的时间是50287毫秒 --保持SqlConnection
            //       普通方式插入100000条数据所用的时间是49693毫秒  --一条一条插入

            //使用SqlBulkCopy插入100000条数据所用的时间是1257毫秒
            //       使用事务插入100000条数据所用的时间是5870毫秒  --使用事务插入数据,分组执行
            //       使用事务插入100000条数据所用的时间是25062毫秒  --先存到string再一次性执行
            //       使用事务插入100000条数据所用的时间是16943毫秒  --在事务里一条一条执行
            //       普通方式插入100000条数据所用的时间是55764毫秒 --保持SqlConnection
            //       普通方式插入100000条数据所用的时间是58620毫秒  --一条一条插入

            //使用SqlBulkCopy插入100000条数据所用的时间是1409毫秒
            //       使用事务插入100000条数据所用的时间是6041毫秒  --使用事务插入数据,分组执行
            //       使用事务插入100000条数据所用的时间是23330毫秒  --先存到string再一次性执行
            //       使用事务插入100000条数据所用的时间是15313毫秒  --在事务里一条一条执行
            //       普通方式插入100000条数据所用的时间是50665毫秒 --保持SqlConnection
            //       普通方式插入100000条数据所用的时间是50775毫秒  --一条一条插入

            //通过三次测试分析发现,SqlBulkCopy速度是最快的,其次是SQL语句组合分组后执行,然后是SQL事务一条一条执行,然后是其它的
            //速度比是1:5:15:50,推荐前两种方法

            Console.ReadLine();

        }

        /// <summary>
        /// 使用普通插入数据
        /// </summary>
        /// <returns></returns>
        private static long CommonInsert()
        {
            Stopwatch stopwatch = new Stopwatch();
            stopwatch.Start();
            for (int i = 0; i < count; i++)
            {
                SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "insert into passport(PassportKey) values('" + Guid.NewGuid() + "')");
            }
            stopwatch.Stop();
            return stopwatch.ElapsedMilliseconds;
        }


        /// <summary>
        /// 使用普通插入数据,保持SqlConnection
        /// </summary>
        /// <returns></returns>
        private static long CommonInsert1()
        {
            Stopwatch stopwatch = new Stopwatch();
            stopwatch.Start();
            SqlConnection sqlconn = new SqlConnection(SqlHelper.SqlConnection);
            for (int i = 0; i < count; i++)
            {
                SqlHelper.ExecuteNonQuery(sqlconn, CommandType.Text, "insert into passport(PassportKey) values('" + Guid.NewGuid() + "')");
            }
            stopwatch.Stop();
            return stopwatch.ElapsedMilliseconds;
        }


        /// <summary>
        /// 使用事务插入数据
        /// </summary>
        /// <returns></returns>
        private static long TransactionInsert()
        {
            Stopwatch stopwatch = new Stopwatch();
            stopwatch.Start();

            using (SqlConnection ConnNow = new SqlConnection(SqlHelper.SqlConnection))
            {
                ConnNow.Open();
                using (SqlTransaction trans = ConnNow.BeginTransaction())
                {
                    try
                    {
                        for (int i = 0; i < count; i++)
                        {
                            SqlHelper.ExecuteNonQuery(trans, CommandType.Text, "insert into passport(PassportKey) values('" + Guid.NewGuid() + "')");
                        }
                        trans.Commit();
                    }
                    catch (Exception ex)
                    {
                        trans.Rollback();
                    }
                }
            }

            stopwatch.Stop();
            return stopwatch.ElapsedMilliseconds;
        }

        /// <summary>
        /// 使用事务插入数据,先存到string再一次性执行
        /// </summary>
        /// <returns></returns>
        private static long TransactionInsert1()
        {
            Stopwatch stopwatch = new Stopwatch();
            stopwatch.Start();
            StringBuilder sb = new StringBuilder();
            for (int i = 0; i < count; i++)
            {
                sb.AppendFormat("insert into passport(PassportKey) values('{0}');", Guid.NewGuid());
            }
            using (SqlConnection ConnNow = new SqlConnection(SqlHelper.SqlConnection))
            {
                ConnNow.Open();
                using (SqlTransaction trans = ConnNow.BeginTransaction())
                {
                    try
                    {
                        SqlHelper.ExecuteNonQuery(trans, CommandType.Text, sb.ToString());
                        for (int i = 0; i < count; i++)
                        {
                            SqlHelper.ExecuteNonQuery(trans, CommandType.Text, "insert into passport(PassportKey) values('" + Guid.NewGuid() + "')");
                        }
                        trans.Commit();
                    }
                    catch (Exception ex)
                    {
                        trans.Rollback();
                    }
                }
            }

            stopwatch.Stop();
            return stopwatch.ElapsedMilliseconds;
        }

        /// <summary>
        /// 使用事务插入数据,分组执行-- 感谢 闫道民 提供
        /// </summary>
        /// <returns></returns>
        private static long TransactionInsert2()
        {
            Stopwatch stopwatch = new Stopwatch();
            stopwatch.Start();
            using (SqlConnection ConnNow = new SqlConnection(SqlHelper.SqlConnection))
            {
                ConnNow.Open();
                string tmp = string.Empty; ;
                int itmp = 0;
                using (SqlTransaction trans = ConnNow.BeginTransaction())
                {
                    try
                    {
                        for (int i = 0; i < count; i++)
                        {
                            itmp++;
                            tmp = tmp + "insert into passport(PassportKey) values('" + Guid.NewGuid() + "');";
                            if (itmp == 180)
                            {
                                itmp = 0;
                                SqlHelper.ExecuteNonQuery(trans, CommandType.Text, tmp);
                                tmp = string.Empty;
                            }
                        }
                        SqlHelper.ExecuteNonQuery(trans, CommandType.Text, tmp);
                        trans.Commit();
                    }
                    catch (Exception ex)
                    {
                        trans.Rollback();
                    }
                }
            }

            stopwatch.Stop();
            return stopwatch.ElapsedMilliseconds;
        }

        /// <summary>
        /// 使用SqlBulkCopy方式插入数据
        /// </summary>
        /// <returns></returns>
        private static long SqlBulkCopyInsert()
        {
            Stopwatch stopwatch = new Stopwatch();
            stopwatch.Start();

            DataTable dataTable = GetTableSchema();
            for (int i = 0; i < count; i++)
            {
                DataRow dataRow = dataTable.NewRow();
                dataRow[2] = Guid.NewGuid();
                dataTable.Rows.Add(dataRow);
            }

            //Console.WriteLine(stopwatch.ElapsedMilliseconds);//初始化数据时间

            SqlBulkCopy sqlBulkCopy = new SqlBulkCopy(SqlHelper.SqlConnection);
            sqlBulkCopy.DestinationTableName = "Passport";

            if (dataTable != null && dataTable.Rows.Count != 0)
            {
                sqlBulkCopy.WriteToServer(dataTable);
            }
            sqlBulkCopy.Close();


            stopwatch.Stop();
            return stopwatch.ElapsedMilliseconds;
        }


        private static DataTable GetTableSchema()
        {
            return SqlHelper.ExecuteDataset(SqlHelper.SqlConnection, CommandType.Text, "select * from Passport where 1=2").Tables[0];
        }

    }
}



cnblogs:SQLServer中批量插入数据方式的性能对比

 


引用本页地址:http://www.yongfa365.com/item/SqlBulkCopy.html
 
 
相关链接
 
网友评论:
1 闫道民 - 2010-5-12 16:56:30
你好,柳永法;
SqlHelper.ExecuteNonQuery(SqlHelper.SqlConnection, CommandType.Text, "insert into passport(PassportKey) values('" + Guid.NewGuid() + "')");
这行代码的SqlHelper.SqlConnection应该是字符串吧,也就是ConnectionString;
如果是这样做,每次执行SQL语句都会创建类型为SqlConnection的对象,并且连接数据库,时间必然会很长。
如果先创建数据库连接SqlConnection objConnection,
再调用SqlHelper.ExecuteNonQuery(objConnection,...)方法,结果应该与SqlBulkCopy一样快吧。
如果有问题,请MAIL:ydmdzq_helloworld@hotmail.com
 
2 柳永法 - 2010-5-12 18:25:02

@闫道民

不是你说的那样,他会使用连接池,速度即便慢也不会慢很多,参看:


ADO.NET 2.0技术内幕 之 连接池


另,我亲自都测试过了,换你的说法我的测试结果是:


使用SqlBulkCopy插入1000000条数据所用的时间是29069毫秒


              普通方式插入1000000条数据所用的时间是634071毫秒

 
3 匿名网友 - 2010-9-7 16:38:17
good,good,good
 
4 SP - 2011-1-25 15:30:27
这样做的话只能适用于同样表结构的数据库中传输数据,而且不能有关系之类约束。其实就是日志传输,把数据库追加到数据库日志之后
 
5 李昭东 - 2011-4-6 18:43:03
下载了,学习一下。应该是很强大的,谢谢分享。
 
6 Black - 2011-4-13 1:16:29
这个东西我也看过相关介绍,感觉对于程序员来说作用不是很大,现在的客户对于excel文件数据的导入需求差别太大,对于一些特殊字段的是否导入都有很明确的要求,况且程序员一般在开发数据库实例时,对于数据表结构设计时,为了适应程序需要,往往会增加很多和客户业务不相关而在程序开发过程中需要使用到的一些辅助字段(比如标记,状态之类),这些字段有些只是在程序后台使用,有些字段甚至在客户使用系统过程中都不会知道有这个东西,如果只是程序员或者测试员在开发或测试过程中为了快速导入测试数据来使用这个功能貌似也很鸡肋,sql自带的数据导入和导出功能就可以满足需求了,说了这么多,我还是没看出楼主推荐它的好处。。。
 
姓名: 记住我
网址:
邮箱:
内容:
验证码:  验证码图片 看不清? 换张图试试
 
     
 
 
文章分类
 
 
.Net + C#(73)
 
 
ASP+VBS(161)
 
 
 
Linux(10)
 
 
 
web 2.0(26)
 
 
 
 
 
心程(68)
 
生活(97)
 
 
     

Power by :柳永法(yongfa365)'Blog  | 京ICP备07011491号  QQ:64049027  E-mail:64049027@qq.com yongfa365'CodePlex yongfa365'CodeGoogle

申请友情链接 要求:跟本站主题相类似正规网站,双方交换为首页位置

转载请注明来源,以便后人及时得到最新、修正、加强版!!!