【转】C#使用DataSet Datatable更新数据库的三种实现方法

转自:http://www.jb51.net/article/54418.htm

这篇文章主要介绍了C#使用DataSet Datatable更新数据库的三种实现方法,需要的朋友可以参考下
id="cproIframe_u1892994_2" height="90" marginheight="0" src="http://pos.baidu.com/acom?adn=3&at=231&aurl=&cad=1&ccd=32&cec=gb2312&cfv=16&ch=0&col=zh-CN&conBW=0&conOP=1&cpa=1&dai=2&dis=0&ltr=http%3A%2F%2Fwww.baidu.com%2Fs%3Fie%3Dutf-8%26f%3D8%26rsv_bp%3D1%26rsv_idx%3D2%26ch%3D%26tn%3D95171689_hao_pg%26bar%3D%26wd%3DC%2523%25E5%25A6%2582%25E4%25BD%2595%25E4%25BF%25AE%25E6%2594%25B9dataset%25E6%259F%2590%25E4%25B8%25AA%25E5%25AD%2597%25E6%25AE%25B5%25E7%259A%2584%25E5%2580%25BC%25E5%25B9%25B6%25E5%2590%258C%25E6%25AD%25A5%25E6%2595%25B0%25E6%258D%25AE%25E5%25BA%2593%26rsv_spt%3D1%26rsv_pq%3De400823600003095%26rsv_t%3D2c60OLcXAozSKdMzk6IYqrya2pfvnVzMDqWfzOGuJfeegBdhadHtJf4mSW4iq1Ak4Cbi%26rsv_enter%3D1%26inputT%3D13837&ltu=http%3A%2F%2Fwww.jb51.net%2Farticle%2F54418.htm&lu_161=0&lunum=6&n=jb51_cpr&pcs=1542x709&pis=10000x10000&ps=489x304&psr=1600x900&pss=1542x492&qn=8aae99581849e284&rad=&rsi0=580&rsi1=90&rsi5=4&rss0=%23FFFFFF&rss1=%23F7FCFF&rss2=%230000ff&rss3=%23444444&rss4=%23008000&rss5=&rss6=%23e10900&rss7=&scale=&skin=tabcloud_skin_3&stid=5&td_id=1892994&titFF=%E5%AE%8B%E4%BD%93&titFS=12&titTA=left&tn=text_default_580_90&tpr=1425536670777&ts=1&version=2.0&xuanting=0&dtm=BAIDU_DUP2_SETJSONADSLOT&dc=2&di=u1892994&tt=1425536670303.935.1144.1144" frameborder="0" width="580" allowtransparency="" marginwidth="0" scrolling="no">

本文以实例形式讲述了使用DataSet Datatable更新数据库的三种实现方法,包括CommandBuilder 方法、DataAdapter 更新数据源以及使用sql语句更新。分享给大家供大家参考之用。具体方法如下:

一、自动生成命令的条件 CommandBuilder 方法

a)动态指定 SelectCommand 属性

b)利用 CommandBuilder 对象自动生成 DataAdapter 的 DeleteCommand、InsertCommand 和 UpdateCommand。

c)为了返回构造 INSERT、UPDATE 和 DELETE 。SQL CommandBuilder 必须执行 SelectCommand。

即:必须额外经历一次到数据源的行程,这可能会降低性能。这也是自动生成命令的缺点。

d)SelectCommand 还必须返回至少一个主键或唯一列.

当CommandBuilder和DataAdapter关联时,就会自动生成DeleteCommand、InsertCommand 和 UpdateCommand中为空的命令。即不空的不生成。

e)必须是一个表,SELECT的不能是多个表的联合。

动生成命令的规则:

在数据源处为表中所有 RowState 为 Added 的行插入一行(不包括标识、表达式或时间戳等列)。
为 Modified 的行更新行(列值匹配行的主键列值) 。
Deleted 的行删除行(列值匹配行的主键列值).这就是为什么要求条件c.d

注意:

a)因为从SELECT数据到UPDATE数据,中间这段时间有可能别的用户已经对数据进行了修改。自动生成命令这种UPDATE只对在行包含所有原始值并且尚未从数据源中删除时更新。

b)自动命令生成逻辑为独立表生成 INSERT、UPDATE 或 DELETE 语句,而不考虑与数据源中其他表的任何关系。因此,当调用 Update 来为参与数据库中外键约束的列提交更改时,可能会失败。若要避免这一异常,请不要使用 CommandBuilder 来更新参与外键约束的列,而应显式地指定用于执行该操作的语句。

下面是自动生成命令的例子

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// Assumes that connection is a valid SqlConnection object.
SqlDataAdapter adapter = new SqlDataAdapter(
"SELECT * FROM dbo.Customers" , connection);
SqlCommandBuilder builder = new SqlCommandBuilder(adapter);
builder.QuotePrefix = "[" ;
builder.QuoteSuffix = "]" ;
  
DataSet custDS = new DataSet();
  
connection.Open();
adapter.Fill(custDS, "Customers" );
  
// Code to modify data in the DataSet here.
  
// Without the SqlCommandBuilder, this line would fail.
adapter.Update(custDS, "Customers" );
connection.Close();

二、使用 DataAdapter 更新数据源

需要注意:

a)如果 SelectCommand 返回 OUTER JOIN 的结果,则 DataAdapter 不会为生成的 DataTable 设置 PrimaryKey 值。您必须自己定义PrimaryKey 以确保正确解析重复行.

b)如果对 DataSet、DataTable 或 DataRow 调用 AcceptChanges,则将使 DataRow 的所有 Original 值都将被重写为该 DataRow 的 Current 值。如果已修改将该行标识为唯一行的字段值,那么当调用 AcceptChanges 后,Original 值将不再匹配数据源中的值。

看看下面例子:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
// Assumes connection is a valid SqlConnection.
SqlDataAdapter dataAdpater = new SqlDataAdapter(
"SELECT CategoryID, CategoryName FROM Categories" , connection);
  
dataAdpater.UpdateCommand = new SqlCommand(
"UPDATE Categories SET CategoryName = @CategoryName " +
"WHERE CategoryID = @CategoryID" , connection);
  
dataAdpater.UpdateCommand.Parameters.Add(
"@CategoryName" , SqlDbType.NVarChar, 15, "CategoryName" );
  
SqlParameter parameter = dataAdpater.UpdateCommand.Parameters.Add(
"@CategoryID" , SqlDbType.Int);
parameter.SourceColumn = "CategoryID" ;
parameter.SourceVersion = DataRowVersion.Original;
  
DataSet dataSet = new DataSet();
dataAdpater.Fill(dataSet, "Categories" );
  
DataRow row = dataSet.Tables[ "Categories" ].Rows[0];
row [ "CategoryName" ] = "New Category" ;
  
dataAdpater.Update(dataSet, "Categories" );

插入、更新和删除的排序

在许多情况下,以何种顺序向数据源发送通过 DataSet 作出的更改是相当重要的。
例如,如果已更新现有行的主键值并且添加了具有新主键值的新行,则务必要在处理插入之前处理更新。

可以使用 DataTable 的 Select 方法来返回仅引用具有特定 RowState 的 DataRow 数组。然后可以将返回的 DataRow 数组传递到 DataAdapter 的 Update 方法来处理已修改的行。通过指定要更新的行的子集,可以控制处理插入、更新和删除的顺序。

示例如下:

?
1
2
3
4
5
6
7
8
DataTable table = dataSet.Tables[ "Customers" ];
// First process deletes.
adapter.Update(table.Select( null , null , DataViewRowState.Deleted));
// Next process updates.
adapter.Update(table.Select( null , null
DataViewRowState.ModifiedCurrent));
// Finally, process inserts.
adapter.Update(table.Select( null , null , DataViewRowState.Added));

三、使用sql语句更新

例如:

?
1
2
3
4
5
6
7
8
9
10
cmd = new OleDbCommand( string .Format( @"insert into worker(workerid,workername,password,phoneno) values ('{0}','{1}','{2}','{3}') " , textBox1.Text, textBox2.Text, textBox3.Text, textBox4.Text),oc);
oc.Open();
try
{
  int i = cmd.ExecuteNonQuery();
}
catch (Exception ex)
{
  
}

性能的优劣及使用的情形,还未完全明白。

一般的,绑定bindingsource,用datatable绑定bindingsource (实质上绑定的是datatable。defaultview,同时可用到dataview的筛选功能,但是在筛选完后,filter要重置为null,否则出现的一直是经过筛选的数据)

其他:

使用builder 的作用:

?
1
OleDbCommandBuilder cb = new OleDbCommandBuilder(da);

这个主要是为了让C#自动为OleDbDataAdapter da生成相对应的DeleteCommand,UpdateCommand!

希望本文所述对大家的C#数据库程序设计有所帮助。


  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值