DataSet 的 Merge 方法

简介:

是MSDN中对Merge方法使用说明:

Merge 方法用于合并架构大致相似的两个 DataSet 对象。合并在客户端应用程序上通常用于将数据源中最近的更改合并到现有的 DataSet 中。这使客户端应用程序能够拥有用数据源中的最新数据刷新的 DataSet。通常在一系列过程的末尾调用 Merge 方法,这些过程涉及验证更改、消除错误、使用更改更新数据源并最后刷新现有的 DataSet。 在客户端应用程序中,通常有这样一个按钮,用户可以单击它来收集已更改的数据并对其进行验证,然后将其发送回中间层组件。在这种情况下,将首先调用 GetChanges 方法。该方法返回另一个为验证和合并而优化的 DataSet。第二个 DataSet 对象只包含已更改的 DataTable 和 DataRow 对象,结果产生初始 DataSet 的子集。该子集通常较小,因此可以更有效率地传递回中间层组件。然后,中间层组件将通过存储过程使用更改更新初始数据源。然后,中间层可以发送回一个新的 DataSet,其中包含数据源中的初始数据和最新数据(通过再次运行初始查询);或者它可以发送回包含从数据源对其进行的所有更改的子集。(例如,如果数据源自动创建唯一主键值,则可以将这些值传播回客户端应用程序。)在哪一种情况下都可以使用 Merge 方法将返回的 DataSet 合并回客户端应用程序的初始 DataSet。 当将新的源 DataSet 合并到目标中时,DataRowState 值为 Unchanged、Modified 或 Deleted 的任何源行都会与具有同一主键值的目标行相匹配。DataRowState 值为 Added 的源行将匹配主键值与新源行相同的新目标行。

根据以上说明,我们知道Merge方法在合并两个数据集时,是以行的主键值为主要对比参照。这样在向数据集添加新行时不会有任何问题,在修改了行且不修改主键值的情况下也不会有问题,但是在更改行时如果你修改了主键的值,那问题就来了…… 下面我们就举例说明:

SQL Server下的一个Products表结构如下:

 

SQL Server下的一个Products表结构如下:

 

列名

数据类型

说明

Code

nchar

产品代码(主键列)

Name

nvarchar

产名名称

UnitPrice

numeric

产品单价

 

.NET中使用XSD生成一个对应的ProductsData.xsd结构如下:

 

列名

数据类型

说明

Code

string

产品代码(主键列)

Name

string

产名名称

UnitPrice

decimal

产品单价

 

根据MSDN的说明,我们在前台通过ProductsData添加或修改数据后在提交后台更新时,通常做法如下:

 

        // 创建一个新数据集来保存对主数据集所做的更改

 

        ProductsData dataSetChanges;

 

        dataSetChanges = (ProductsData)(productsData.GetChanges());

 

        // 检查是否做了任何更改

 

        if(dataSetChanges != null) {

 

            try {

 

                // 需要做一些更改,所以尝试通过调用 update 方法

 

// 和传递数据集以及任何参数来更新数据源

 

                UpdateDataSource(dataSetChanges);

 

                productsData.Merge(dataSetChanges);

 

                productsData.AcceptChanges();

 

            }

 

            catch (System.Exception eUpdate) {

 

                throw eUpdate;

 

            }

 

        }

 

以上代码是根据VS.NET的数据窗体生成向导写的,依据以上代码我们模拟向数据集添加一行数据并更新后的情况:

 

Code

Name

UnitPrice

1001

金砂朱古力

120.00

 

没问题,下面我们修改这行数据再更新,这里我们把Code改为1002,更新之后结果数据并没有按我们预想的把原本行Code列的值1001改为1002,而是添加了一行:

 

Code

Name

UnitPrice

1001

金砂朱古力

120.00

1002

金砂朱古力

130.00

 

注:通常情况下我们是很少更改主键值,但在代码没有被使用的情况下,一般是允许更改Code的,特别是在系统实施的阶段。

 

出现以上问题的原因其实不奇怪,按Merge方法的原理,这一点也没错,但这是我们不希望的结果,怎么解决呢,难道不允许用户更改主键,但好象不符合实际。怎么解决呢?

 

既然要用Merge方法,那只有依Merge合并数据的原理去做,不让改主键我们就不改主键,我们可以给前台的ProductsData的数据集加多一个额外的主键ID列,并把它的AutoIncrement设为true,将原本的主键列Code改为Key列,至于后台SQL Server中的源表不作任何修改,修改后如下:

 

列名

数据类型

说明

ID

Int

自动增长列 (主键)

Code

String

产品代码   (key 键)

Name

String

产名名称

UnitPrice

decimal

产品单价

 

经这样一改,由于ID是自动一个自递增列,我们并不去修改它的值,这样我们就可以随意更改 Code 列的值了,Merge 方法在合并数据时由于是依据ID例进行比对所以也不会再出现前面加多一行的问题了。

分类:  ASP.NET
本文转自左正博客园博客,原文链接:http://www.cnblogs.com/soundcode/archive/2013/01/16/2862273.html ,如需转载请自行联系原作者
相关文章
【Stata】append和merge的区别
【Stata】append和merge的区别
513 0
Pandas pd.merge() 报错:ValueError: You are trying to merge on int64 and object columns.
Pandas pd.merge() 报错:ValueError: You are trying to merge on int64 and object columns.
Pandas pd.merge() 报错:ValueError: You are trying to merge on int64 and object columns.
|
SQL 分布式计算 关系型数据库
Dataset 和DataFrame 的区别_Row 对象 | 学习笔记
快速学习 Dataset 和DataFrame 的区别_Row 对象
280 0
Dataset 和DataFrame 的区别_Row 对象 | 学习笔记
|
SQL 分布式计算 大数据
Dataset介绍_将 Dataset 转为同泛型的 RDD | 学习笔记
快速学习 Dataset介绍_将 Dataset 转为同泛型的 RDD
176 0
Dataset介绍_将 Dataset 转为同泛型的 RDD | 学习笔记
|
JavaScript Python
shuffle() 方法、removeAttribute() 方法、split() 方法
内容包括demo代码,应用和定义,以及参考文献,本文主要内容是学习js期间学到的一些实用的零碎的js1知识,我都记下来了,需要的朋友可以过来参考下,前后可能没有太大的相关性。喜欢的可以点个赞,希望对大家有所帮助。 参数lst -- 这可能是一个列表或元组。作用:打乱一个有序数组,最高效的数组乱序方法应用:打乱输出后,从头输出可用来输出随机数字。 这里有篇文章介绍数组乱序效率www.jb51.net/article/560… HTML DOM removeAttribute()
161 0
shuffle() 方法、removeAttribute() 方法、split() 方法
|
Python
使用Numpy将数据集中的data和target同时shuffle
假设现在有图像数据imgs和对应标签targets。数据维度分别如下 imgs.shape = (num, channel, width, height) targets.shape = (num, class) 因为通常我们需要将数据打散,这样的好处是可以让模型训练更具鲁棒性,那么如何同时打散da...
1394 0
|
SQL 关系型数据库 Oracle
|
XML 缓存 数据库
DataSet用法详细 转
一、特点介绍1、处理脱机数据,在多层应用程序中很有用。2、可以在任何时候查看DataSet中任意行的内容,允许修改查询结果的方法。3、处理分级数据4、缓存更改5、XML的完整性:DataSet对象和XML文档几乎是可互换的。
903 0