最近公司需要优化导入的问题,由于之前使用的方式是生成 Insert 语句插入数据库,数据量小的时候还行,但是随着发展数据量渐渐大了,之前的方法性能就跟不上了,于是发现了 SqlBulkCopy 这个类。

使用 SqlBulkCopy 类只能向 SQL Server 表写入数据。但是,数据源不限于 SQL Server;可以使用任何数据源,只要数据可加载到 DataTable 实例或可使用 IDataReader 实例读取数据。

public class Conn
{
private static string StrConn
{
get
{
return ConfigurationManager.ConnectionStrings["StrConn"].ToString();
//return ConfigurationManager.AppSettings["StrConn"].ToString();
}
} public static SqlConnection SqlConn
{
get
{
return new SqlConnection(StrConn);
}
}
} public class SqlHelper
{ public DataTable GetDataTable(string sql)
{
DataTable dt = new DataTable();
SqlConnection conn = null;
SqlDataAdapter sda = null;
try
{
conn = Conn.SqlConn;
sda = new SqlDataAdapter(sql, conn);
conn.Open();
sda.Fill(dt);
}
catch (Exception ex)
{ }
finally
{
if (conn != null)
{
conn.Close();
conn.Dispose();
}
if (sda != null)
{
sda.Dispose();
}
}
return dt;
} public DataSet GetDataSet(string sql)
{
DataSet ds = new DataSet();
SqlConnection conn = null;
SqlDataAdapter sda = null;
try
{
conn = Conn.SqlConn;
sda = new SqlDataAdapter(sql, conn);
conn.Open();
sda.Fill(ds);
}
catch (Exception ex)
{ }
finally
{
if (conn != null)
{
conn.Close();
conn.Dispose();
}
if (sda != null)
{
sda.Dispose();
}
}
return ds;
} /// <summary>
/// 使用事务插入方法
/// </summary>
/// <param name="dt">源数据</param>
/// <param name="tableName">目标表名</param>
public void InsertO(DataTable dt, string tableName)
{
using (SqlConnection conn = Conn.SqlConn)
{
using (SqlBulkCopy sqlBuleCopy = new SqlBulkCopy(conn.ConnectionString, SqlBulkCopyOptions.CheckConstraints | SqlBulkCopyOptions.Default | SqlBulkCopyOptions.UseInternalTransaction))
{
try
{
//设置目标表名,即数据库表名
sqlBuleCopy.DestinationTableName = tableName; //设置每一批次的行数,即达到指定的行数就插入一次数据库
sqlBuleCopy.BatchSize = ; //设置超时之前完成的时间(秒)
sqlBuleCopy.BulkCopyTimeout = ;
for (int i = ; i < dt.Columns.Count; i++)
{
//设置源数据列与目标表的列的映射关系,第一个参数为源数据列,第二个参数为目标表列
sqlBuleCopy.ColumnMappings.Add(dt.Columns[i].ColumnName, dt.Columns[i].ColumnName);
}
sqlBuleCopy.WriteToServer(dt);
}
catch (Exception)
{
}
}
}
} /// <summary>
/// 未使用事务插入方法
/// </summary>
/// <param name="dt">源数据</param>
/// <param name="tableName">目标表名</param>
public void InsertT(DataTable dt, string tableName)
{
using (SqlConnection conn = Conn.SqlConn)
{
using (SqlBulkCopy sqlBuleCopy = new SqlBulkCopy(conn))
{
try
{
conn.Open();
//设置目标表名,即数据库表名
sqlBuleCopy.DestinationTableName = tableName; //设置每一批次的行数,即达到指定的行数就插入一次数据库
sqlBuleCopy.BatchSize = ; //设置超时之前完成的时间(秒)
sqlBuleCopy.BulkCopyTimeout = ;
for (int i = ; i < dt.Columns.Count; i++)
{
//设置源数据列与目标表的列的映射关系,第一个参数为源数据列,第二个参数为目标表列
sqlBuleCopy.ColumnMappings.Add(dt.Columns[i].ColumnName, dt.Columns[i].ColumnName);
}
sqlBuleCopy.WriteToServer(dt);
}
catch (Exception)
{
conn.Close();
conn.Dispose();
}
finally
{
conn.Close();
conn.Dispose();
}
}
}
}
}

我的源数据是使用 Excel 导入的数据,导入的方法就不说了,不是这里的重点,之后我会专门总结一下 Excel 导入的方法。然后查询目标表需要插入数据的字段,修改源数据表的字段名和类型,然后调用批量插入的方法。

protected void btnImport_Click(object sender, EventArgs e)
{
try
{
//获取导入的数据
DataSet ds = BI.ExecleDs(savePath, "");
if (ds != null && ds.Tables.Count > )
{
DataTable dt = ds.Tables[]; //查询目标表需要插入的字段
string sql = " select U_No,U_Name,U_Pwd,P_Id from UserInfo ";
DataTable dt1 = sqlhelper.GetDataTable(sql); if (dt1 != null)
{
for (int i = ; i < dt1.Columns.Count; i++)
{
//修改源数据表的字段类型和字段名称
dt.Columns[i].DataType = dt1.Columns[i].DataType;
dt.Columns[i].ColumnMapping = dt1.Columns[i].ColumnMapping;
dt.Columns[i].ColumnName = dt1.Columns[i].ColumnName;
} sqlhelper.InsertO(dt, "UserInfo");
}
}
}
catch (Exception ex)
{ throw;
}
}

以上这种修改数据类型的方法,如果碰到数据类型不一致并且 DataTable 有数据的时候,会报错,不能修改有数据的列的数据类型。(好像是废话,嘿嘿,没有数据和有数据时不能修改数据类型,这完全是没用的。)

所以就有了下面的方法,先实例化一个新的 DataTable,然后复制目标表的架构,然后再把数据保存到新的 DataTable 中。

protected void btnImport_Click(object sender, EventArgs e)
{
try
{
//获取导入的数据
DataSet ds = BI.ExecleDs(savePath, "");
if (ds != null && ds.Tables.Count > )
{
DataTable dt = ds.Tables[]; //查询目标表需要插入的字段
string sql = " select U_No,U_Name,U_Pwd,P_Id from UserInfo ";
DataTable dt1 = sqlhelper.GetDataTable(sql);
DataTable dt2 = new DataTable(); if (dt1 != null)
{
//复制目标表的架构
dt2 = dt1.Clone(); for (int i = ; i < dt1.Rows.Count; i++)
{
DataRow dr = dt2.NewRow();
dr = dt1.Rows[i];
dt2.Rows.Add(dr.ItemArray);
} sqlhelper.InsertO(dt2, "UserInfo");
}
}
}
catch (Exception ex)
{ throw;
}
}

如果源数据表的列和目标表的列的顺序或列名不相同,那就必须使用 ColumnMappings.Add() 方法设置映射关系。

参考:

http://www.cnblogs.com/zfanlong1314/archive/2013/02/05/2892998.html

C# 使用 SqlBulkCopy 类批量复制数据到数据库的更多相关文章

  1. 用.net中的SqlBulkCopy类批量复制数据 (转载)

    在软件开发中,把数据从一个地方复制到另一个地方是一个普遍的应用. 在很多不同的场合都会执行这个操作,包括旧系统到新系统的移植,从不同的数据库备份数据和收集数据. .NET 2.0有一个SqlBulkC ...

  2. 使用asp.net 2.0中的SqlBulkCopy类批量复制数据

    介绍:在软件开发中,把数据从一个地方复制到另一个地方是一个普遍的应用. 在很多不同的场合都会执行这个操作,包括旧系统到新系统的移植,从不同的数据库备份数据和收集数据. ASP.NET 2.0有一个Sq ...

  3. C# 使用SqlBulkCopy类批量复制大数据

    用途说明: 前些日子,公司要求做一个数据导入程序,要求将Excel数据,大批量的导入到数据库中,尽量少的访问数据库,高性能的对数据库进行存储.于是在网上进行查找,发现了一个比较好的解决方案,就是采用S ...

  4. C# SqlBulkCopy类批量导入数据

    特别注意  sqlbulkcopy.ColumnMappings.Add(dt.Columns[i].ColumnName, dt.Columns[i].ColumnName); 插入的时候列的顺序可 ...

  5. 使用SqlBulkCopy类批量复制大数据

    using System; using System.Configuration; using System.Data; using System.Data.SqlClient; using Syst ...

  6. SqlBulkCopy 批量复制数据到数据库

    1.简介 1.MSDN 核心方法:SqlBulkCopy.WriteToServer 将所有行从数据源复制到 SqlBulkCopy 对象的 DestinationTableName 属性指定的目标表 ...

  7. SqlBulkCopy类进行大数据(一万条以上)插入测试

    好多天没写博客了,刚刚毕业一个多月! 关于上一篇博客中提到的,在进行批量数据插入数据库的时候可以通过给存储过程传递一个类型为Table的参数进行相关操作,在这个过程中本人没有进行效率的测试.后来查找发 ...

  8. SqlBulkCopy类进行大数据(10000万条以上)插入测试

    好多天没写博客了,刚刚毕业一个多月,在IT的路上真是迷茫啊! 关于上一篇博客中提到的,在进行批量数据插入数据库的时候可以通过给存储过程传递一个类型为Table的参数进行相关操作,在这个过程中本人没有进 ...

  9. 转:SqlBulkCopy类进行大数据(一万条以上)插入测试

    转自:https://www.cnblogs.com/LenLi/p/3903641.html 结合博主实例,自己测试了一下,把数据改为3万行更明显!! 关于上一篇博客中提到的,在进行批量数据插入数据 ...

随机推荐

  1. poj3301--Texas Trip(最小正方形覆盖)

    题目链接:点击打开链接 题目大意:给出n个点的坐标.如今要求一个正方形,全然包围n个点.而且正方形面积最小,求最小的正方形面积. 表示不能理解为什么面积随着角度的变化是一个单峰的函数,等待大牛告诉一下 ...

  2. jQuery对象与javaScript对象的互换

    1. jQuery对象-->Dom对象 1) 通过 .[0] 的方式:       var $s = $(.class);  (jQuery对象)       var s = $s.[0];   ...

  3. [Openwrt]wifi桥接设置

    1/连接wifi,用扫描加入,新建一个ingerface  client模式 2/新建wifi ,network选lan和自身新建的interface.此interface设置为bridge模式,连接 ...

  4. jQuery制作简洁的多级联动Select下拉框

    今天我们要来分享一款很实用的jQuery插件,它是一个基于jQuery多级联动的省市地区Select下拉框,并且值得一提的是,这款联动下拉框是经过自定义美化过的,外观比浏览器自带的要漂亮许多.另外,这 ...

  5. 学习《深入理解C#》—— 可空类型、可选参数和默认值 (第一章1.3)

    目录 C#可空类型 C# 可选参数和默认值 C# 可空类型 在日常生活中,相信大家都离不开手机,低头族啊!哈哈... 假如手机厂商生产了一款新手机,暂时还未定价,在C#1中我们该怎么做呢? 常见的解决 ...

  6. AWS系列-EC2默认限制说明

    Amazon EC2 提供您可以使用的不同资源,例如实例和卷. 在您创建 AWS 账户时,AWS 会针对每个区域中的这些资源设置限制.此页面列出您在 亚太区域 (东京) 中的 EC2 服务限制. 1. ...

  7. XAML 中使用空格键(空白字符)

    默认情况下,XAML折叠所有空白,这意味着包含空格.tab 键以及回车的长字符串将被转换为单个空格.而且,如果在一个元素内容之前或之后添加空白,这个空格将被完全忽略. 有时这并不是所期望的结果.例如, ...

  8. Android5.0新特性-Material Design

    概述 2014年,Google携Android5.X重装归来.全新的UI设计和更加优化的性能,令开发人员眼前一亮 安装和配置Android5.0开发环境 开发Android还得靠AS.下载地址 htt ...

  9. cocos2dx游戏--三国关羽传【角色扮演类】Demo的制作及实现

    项目地址:https://github.com/moonlightpoet/GuanYuZhuan 主要类及其对应效果: MainScene:菜单界面(用于选择不同剧本) StoryScene:故事界 ...

  10. json的循环

    通过for in: var json = {width: '200px', height: '200px', background: 'green'}; var i = ''; for(i in js ...