C#利用SqlDataAdapte对DataTable进行批量数据操作,可以让我们大大简化操作数据的代码量,我们几乎不需要循环和不关心用户到底是新增还是修改,更不用编写新增和修改以及删除的SQL语句,适配器都帮我们在后台进行了很好的处理.

  如果您要通过 SQL Server 存储过程使用 DataAdapter 来编辑或删除数据,请确保不要在存储过程定义中使用 SET NOCOUNT ON。这将使返回的受影响的行数为零,DataAdapter 会将其解释为并发冲突。在许多情况下,以何种顺序向数据源发送通过 DataSet 所做的更改是非常重要的。例如,如果更新了现有行的主键值,并且添加了以新主键值作为外键的新行,则务必要在处理插入之前处理更新。可以使用 DataTable 的 Select 方法来返回仅引用具有特定 RowState 的 DataRow 数组。然后可以将返回的 DataRow 数组传递给 DataAdapter 的 Update 方法来处理已修改的行。通过指定要更新的行的子集,可以控制处理插入、更新和删除的顺序。以下代码确保首先处理表中已删除的行,然后处理已更新的行,然后处理已插入的行。

 DataTable table = dataSet.Tables["Customers"];
//第一步处理删除.
adapter.Update(table.Select(null, null, DataViewRowState.Deleted));
//接着处理更新
adapter.Update(table.Select(null, null,
DataViewRowState.ModifiedCurrent));
//最后处理新增
adapter.Update(table.Select(null, null, DataViewRowState.Added));

注意   对 DataSet、DataTable 或 DataRow 调用 AcceptChanges 将导致 DataRow 的所有 Original 值被 DataRow 的 Current 值覆盖。如果修改了唯一标识该行的字段值,则在调用 AcceptChanges 后,Original 值将不再匹配数据源中的值。在调用 DataAdapter 的 Update 方法期间会对每一行自动调用 AcceptChanges。在调用 Update 方法期间,通过先将 DataAdapter 的 AcceptChangesDuringUpdate 属性设置为 false,或为 RowUpdated 事件创建一个事件处理程序并将 Status 设置为 SkipCurrentRow,可以保留原始值。(来自MSDN) 

1 强类型数据集

  数据库的表设计好后,可以用VS连接数据库,并将myUser 表拖放到数据集设计器中(重命名为dsUser),默认情况下,数据集会自动生成Fill方法,我们这里需要手动进行删除,只保留表结构即可,数据查询和操作逻辑我们进行自定义.如下图所示:

2 UI设计

  新建一个窗体,编译后可以从工具栏将上一步新建的dsUser强类型数据集拖放到窗体上,另外添加一个BindingSource控件(主要用于对DataGridView中的数据库和DataSet的数据进行绑定)

  BindingSource中选择数据源DataSource为dsUser,然后选择数据集中的一个表myUser进行绑定:

  设置DataGridView的数据源为bindingSource,这样通过bindingSource1就实现了DataGridView和dsUser的数据绑定,也就是说在界面DataGridView上进行操作,可以同步到dsUser中(会在每行上打上标志)

3 数据操作方法

  可以利用SqlDataAdapter批量更新DataTable中的数据库,也支持表格中同时进行了删除/修改和新增的操作,下面的代码中,用select * from {0} where 1=2查询语句为SqlDataAdapter提供架构:

 //生成架构
string selectSQL = string.Format("select * from {0} where 1=2", dt.TableName);
SqlDataAdapter sda = new SqlDataAdapter(selectSQL, ConnectionString)

 而后用SqlCommandBuilder自动构建新增/删除/修改的命令,这样就可以用sda.Update()方法对数据表进行批量操作了:

 SqlCommandBuilder scb  =  new SqlCommandBuilder(sda);
sda.Update(dt);

 完整代码如下:

 public static void UpdateDataSet(DataSet ds, string tableName)
{
try
{
if (tableName == "")
{
throw new ArgumentNullException("tableName 不能为空");
}
//生成架构
string selectSQL = string.Format("select * from {0} where 1=2", tableName);
using (SqlDataAdapter sda = new SqlDataAdapter(selectSQL, ConnectionString))
{ SqlCommandBuilder scb = new SqlCommandBuilder(sda); sda.UpdateCommand = scb.GetUpdateCommand();
sda.InsertCommand = scb.GetInsertCommand();
sda.DeleteCommand = scb.GetDeleteCommand(); sda.Update(ds, tableName);
ds.Tables[tableName].AcceptChanges();
DisposeCommand(sda.UpdateCommand);
DisposeCommand(sda.InsertCommand);
DisposeCommand(sda.DeleteCommand);
sda.Dispose();
scb.Dispose();
}
}
catch (SqlException e)
{
throw e;
}
}

  为了确保在用户编辑完成后,立刻将数据同步到数据集中,必须调用bindingSource的EndEdit方法来同步数据.

 private void dataGridView1_CellEndEdit(object sender, DataGridViewCellEventArgs e)
{
this.bindingSource.EndEdit();
}

  在窗体加载时,调用刷新方法,然后程序会自动将数据绑定到DataGridView上:

 private bool fnRefresh()
{
string sql = string.Format("select * from myUser");
CM.Products.DataAccess.SqlHelper.GetDataTableBySQL(this.dsUser.myUser, sql);
return true;
}

  我们可以通过DataGridView的EndEdit方法来获取当前用户是否已经结束编辑:

 bool isEnd= this.dataGridView1.EndEdit();

  另外可以通过GetChanges()方法来获取修改的数据,返回一个DataTable.

1 DataTable dtModify = this.dsUser.myUser.GetChanges(DataRowState.Modified);

  调用UpdateTable方法,即可对表格进行批量处理(这里就一个表,如果是多个表,那么需要指明表名):

 private bool Save()
{
try
{
DataAccess.SqlHelper.UpdateDataSet(this.dsUser, "myUser");
this.dsUser.myUser.AcceptChanges();//提交数据库
return true;
}
catch (Exception ex)
{
return false;
} }

  当用户选择一个单元格时,会获取到当前行的索引,然后调用RemoveAt()方法进行删除,由于用bindingSource进行了绑定,可以直接调用save()方法进行数据保存:

 private void bindingNavigatorDeleteItem_Click(object sender, EventArgs e)
{
if (this.dataGridView1.CurrentRow != null)
{
int index = this.dataGridView1.CurrentRow.Index;
if (index > -)
{
this.dataGridView1.Rows.RemoveAt(index);
Save();
}
}
}

  当用户单击新增按钮时,我们创建一个强类型的myUserRow,然后给它附上默认值(特别是主键)

 //add row
private void toolStripButton1_Click(object sender, EventArgs e)
{
Common.dsUser.myUserRow dr = this.dsUser.myUser.NewmyUserRow();
dr["ID"] = System.Guid.NewGuid().ToString();
dr["Name"] = "Name";
dr["AddTime"] = DateTime.Now.ToString(); this.dsUser.myUser.AddmyUserRow(dr);
}

C#利用SqlDataAdapte对DataTable进行批量数据操作的更多相关文章

  1. MyBatis的关于批量数据操作的测试

    摘录自:http://www.linuxidc.com/Linux/2012-05/60863.htm MyBatis的前身就是著名的Ibatis,不知何故脱离了Apache改名为MyBatis.My ...

  2. CDH离线数据导入solr:利用MapReduceIndexerTool将json文件批量导入到solr

    场景描述:前段时间,将实时数据通过kafka+flume+morphline的方式接入到solr中.新进来的数据已经可以在solr中看到了,但是以前的历史数据还没有导入solr. CDH提供利用Map ...

  3. 批量插入数据, 将DataTable里的数据批量写入数据库的方法

    大量数据导入操作, 也就是直接将DataTable里的内容写入到数据库 通用方法: 拼接Insert语句, 好土鳖 1. MS Sql Server:   使用SqlBulkCopy 2. MySql ...

  4. java操作Excel之POI(5)利用POI实现使用模板批量导出数据

    后台导出方法: 在源文件夹src下面放个准备好的模板:/com/cy/template/userExportTemplate.xls,这个模板有头部一行: /** * 后台导出方法 * 利用POI实现 ...

  5. SqlBulkCopy将DataTable中的数据批量插入数据库中

    #region 使用SqlBulkCopy将DataTable中的数据批量插入数据库中 /// <summary> /// 注意:DataTable中的列需要与数据库表中的列完全一致.// ...

  6. list转datatable,SqlBulkCopy将DataTable中的数据批量插入数据库

    /// <summary> /// 将泛类型集合List类转换成DataTable /// </summary> /// <param name="list&q ...

  7. 利用Factory-boy和sqlalchemy来批量生成数据库表数据

    测试过程中免不了要构造测试数据,如果是单条数据,还比较简单,但如果是批量数据,就比较麻烦了. 最近看到Factory_boy这个python第三方库,它通过SQLAlchemyModelFactory ...

  8. 利用反射把DataTable自动赋值到Model实体(自动识别数据类型)

    转:http://www.cnblogs.com/the7stroke/archive/2012/04/22/2465591.html using System.Collections.Generic ...

  9. 你好,C++(23) 4.4.2 工资程序成长记:用数组处理批量数据,用循环结构执行重复动作

    4.4  从语句到程序 了解了各种表达式和语句之后,就相当于掌握了写作文要用到的词语和句子,但是,仅有词语和句子是无法构成一篇有意义的文章的.要完成一篇文章,先需要确定这篇文章的结构,是先分述再总述, ...

随机推荐

  1. WP中的语音识别(上):基本识别

    WP 8.1目前许多内容仍处于未确定状态,因此,本文所提及的语音识别,是基于WP8的,在8.1中也差不多,也是使用运行时API来实现,如果大家不知道什么是运行时API,也没关系,不影响学习和开发,因为 ...

  2. Cocos2d-x 3.2 学习笔记(十六)保卫萝卜 游戏主循环与定时器

    保卫萝卜~想法一直存在于想法,实战才是硬道理!有想法就去实现,眼高手低都是空谈.   一.游戏主循环GameSchedule      主循环是游戏处理逻辑,控制游戏进度的地方,处理好主循环是很重要的 ...

  3. Robot Framework自动化测试(七)--- jybot模式

    虽然,很久不用关于Robot Framework框架了,但我这里应该是除了@齐涛-道长之外分享Robot Framework 相关资料比较多的地方了.所以,常常被问到一些关于该框架的问题. 虽然,我一 ...

  4. 可视化(番外篇)——在Eclipse RCP中玩转OpenGL

    最近在看有关Eclipse RCP方面的东西,鉴于Gephi是使用opengl作为绘图引擎,所以,萌生了在Eclipse RCP下添加画布,使用opengl绘图的想法,网上有博文详细介绍这方面的内容, ...

  5. javascript学习—理解addLoadEvent函数

    onload事件是HTML DOM Event 对象的一个属性,又叫事件句柄(Event Handlers),它会在页面或图像加载完成后(注意是加载完成后)立即发生. window.onload = ...

  6. swift3.0的改变

    Swift在这2年的时间内,发展势头迅猛,在它开源后,更是如井喷一样,除了 iOS.mac 平台,还支持了 Linux. 而今年下半年, Swift 3.0 也会随之发布.https://github ...

  7. OpenJudge2728:摘花生 解题报告

    2728:摘花生 总时间限制:  1000ms 内存限制: 65536kB 描述 Hello Kitty 想摘点花生送给她喜欢的米老鼠.她来到一片有网格状道路的矩形花生地(如下图),从西北角进去,东南 ...

  8. Shiro —— 从一个简单的例子开始

    一.Shiro是用来做权限的. 二.权限 1.基本概念: (1)安全实体:要保护的数据. (2)权限:是否有能力去操作(查看.修改.删除 )保护的数据. 2.权限的两个特性 (1)权限的继承性:A 包 ...

  9. 把DataTable转换为泛型List<T>或是JSON

    在开发ASP.NET Web API或ASP.NET MVC时,我们从数据库得到的数据往往是DataSet或是DataTable.为了能让前端JQuery能方便使用至这些数据,我们需要把这些数据转换为 ...

  10. ASP.Net中防止页面刷新重复提交的几种方法

    [摘要] 目前很多网站都要提交页面插入或更新数据库,比如留言本,一个用户提交留言后,如果按F5,就会重新提交一遍留言,导致数据库出现两条一模一样的留言,本文介绍了几种防止页面刷新,导致重复提交数据的方 ...