欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2023-12-21 09:25:58
本文以实例形式讲述了使用dataset datatable更新数据库的三种实现方法,包括commandbuilder 方法、dataadapter 更新数据源以及使用sql...

本文以实例形式讲述了使用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 来更新参与外键约束的列,而应显式地指定用于执行该操作的语句。

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

// 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 值将不再匹配数据源中的值。

看看下面例子:

// 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 方法来处理已修改的行。通过指定要更新的行的子集,可以控制处理插入、更新和删除的顺序。

示例如下:

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语句更新

例如:

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 的作用:

oledbcommandbuilder cb = new oledbcommandbuilder(da);

这个主要是为了让c#自动为oledbdataadapter da生成相对应的deletecommand,updatecommand!

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

上一篇:

下一篇: