ADO.NET中的多数据表操作浅析—读取

ado|数据

ADO.NET中的多数据表操作浅析—读取

作者:郑佐??????? 2004-8-5

???????? 在开发基于.NET平台的数据库应用程序时,我们一般都会用到DataSet,作为ADO.NET的核心类它为我们提供了强大的功能,而整个看上去就像是放在内存内的一个小型数据库,内部包括了DataTable、DataView、DataRow、DataColumn、Constraint以及DataRelation。当初看到它时真的是有点兴奋。

???????? 下面根据我的一些经验来举例说明在ADO.NET中的多表填充、关联表更新以及多个Command对象执行过程中启用事务的操作。欢迎大家交流,或在Blog上留言。

????????

一、准备工作

???????? 对于NorthWind数据库大家都比较熟悉,所以这里拿它为例,我把Customers(客户表)、Orders(订单表)、Order Details(订单详细表)合起来建立了一个类型化的数据集,类型名称为DatasetOrders,每个表只包括一些字段,下面是在Visual Studio .NET中建立的一个截图:

?

图1-1

上面建立了两个关系表示为Customers —> Orders —>Order Details。因为Orders表的OrderID字段为自动增长列,这里把就把它的AutoIncrementSeed和AutoIncrementStep值设置成了-1,这在实际添加订单的过程中可能会比较明显,不过不设也没问题。

????????

二.填充数据集

建立一个窗体程序来演示实际的操作,界面如下:

图2-1

整个应用程序就是一个Form,上面的三个DataGrid分别用来显示相关表的数据,不过他们是互动的。另外的两个单选框用来决定更新数据的方式,两个按钮正如他们的名称来完成相应的功能。

这里我们用一个DataAdapter来完成数据集的填充,执行的存储过程如下:

CREATE PROCEDURE GetCustomerOrdersInfo

AS

SELECT CustomerID,CompanyName,ContactName FROM Customers WHERE CustomerID LIKE 'A%'

?

SELECT OrderID,OrderDate,CustomerID FROM Orders? WHERE CustomerID IN

(SELECT CustomerID FROM Customers WHERE CustomerID LIKE 'A%')

?

SELECT OrderID,ProductID,UnitPrice,Quantity,Discount FROM [Order Details] WHERE OrderID IN

(SELECT OrderID FROM Orders? WHERE CustomerID IN

(SELECT CustomerID FROM Customers WHERE CustomerID LIKE 'A%'))

?

GO

?

为了减少数据量,这里只取了CustomerID以’A’开头的数据。

建立DataAccess类来管理窗体同数据层的交互:

using System;

using System.Data;

using System.Data.SqlClient;

using Microsoft.ApplicationBlocks.Data;

?

namespace WinformTest

{

???? public class DataAccess

???? {

???????? private string _connstring = "data source=(local);initial catalog=Northwind;uid=csharp;pwd=c#.net2004;";

???????? private SqlConnection _conn;

???????? ///构造函数

public DataAccess()

???????? {

????????????? _conn = new SqlConnection(_connstring);

}

下面的函数完成单个数据适配器来完成数据集的填充,

public void FillCustomerOrdersInfo(DatasetOrders ds)

???? ???? {

????????????? SqlCommand comm = new SqlCommand("GetCustomerOrdersInfo",_conn);

????????????? comm.CommandType = CommandType.StoredProcedure;

????????????? SqlDataAdapter dataAdapter = new SqlDataAdapter(comm);

????????????? dataAdapter.TableMappings.Add("Table","Customers");

????????????? dataAdapter.TableMappings.Add("Table1","Orders");

????????????? dataAdapter.TableMappings.Add("Table2","Order Details");

????????????? dataAdapter.Fill(ds);

???????? }

如果使用SqlHelper来填充那就更简单了,

???????? public void FillCustomerOrdersInfoWithSqlHelper(DatasetOrders ds)

???????? {???????????? SqlHelper.FillDataset(_connstring,CommandType.StoredProcedure,"GetCustomerOrdersInfo",ds,new string[]{"Customers","Orders","Order Details"});

???????? }

叉开话题提一下,Data Access Application Block 2.0中的SqlHelper.FillDataset这个方法超过两个表的填充时会出现错误,其实里面的逻辑是错的,只不过两个表的时候刚好凑巧,下面是从里面截的代码:

private static void FillDataset(SqlConnection connection, SqlTransaction transaction, CommandType commandType,

????????????? string commandText, DataSet dataSet, string[] tableNames,

????????????? params SqlParameter[] commandParameters)

???????? {

????????????? if( connection == null ) throw new ArgumentNullException( "connection" );

????????????? if( dataSet == null ) throw new ArgumentNullException( "dataSet" );

????????????? SqlCommand command = new SqlCommand();

????????????? bool mustCloseConnection = false;

????????????? PrepareCommand(command, connection, transaction, commandType, commandText, commandParameters, out mustCloseConnection );

??? ????????? using( SqlDataAdapter dataAdapter = new SqlDataAdapter(command) )

????????????? {

???????? ????????? if (tableNames != null && tableNames.Length > 0)

?????????????????? {

?????????????????????? string tableName = "Table";

?????????????????????? for (int index=0; index < tableNames.Length; index++)

?????????????????????? {

??????????????????????????? if( tableNames[index] == null || tableNames[index].Length == 0 )

???????????????????????????????? throw new ArgumentException( "The tableNames parameter must contain a list of tables, a value was provided as null or empty string.", "tableNames" );

??????????????????????????? tableName += (index + 1).ToString();//这里出现错误

?????????????????????? }

?????????????????? }

?????????????????? dataAdapter.Fill(dataSet);

?????????????????? command.Parameters.Clear();

????????????? }

????????????? if( mustCloseConnection )

?????????????????? connection.Close();

???????? }

?

这里把tableName += (index + 1).ToString();修改成

dataAdapter.TableMappings.Add((index>0)?(tableName+index.ToString()):tableName, tableNames[index]);就能解决问题。

?

接下来看看窗体程序的代码:

public class Form1 : System.Windows.Forms.Form

???? {

???????? private DataAccess _dataAccess;

???????? private DatasetOrders _ds;

???????? //……

???????? //构造函数

???????? public Form1()

???????? {

????????????? InitializeComponent();

????????????? _dataAccess = new DataAccess();

????????????? _ds = new DatasetOrders();

????????????? _ds.EnforceConstraints = false; //关闭约束检查,提高数据填充效率

????????????? this.dataGridCustomers.DataSource = _ds;

????????????? this.dataGridCustomers.DataMember = _ds.Customers.TableName;

????????????? this.dataGridOrders.DataSource = _ds;

????????????? this.dataGridOrders.DataMember = _ds.Customers.TableName+"."+_ds.Customers.ChildRelations[0].RelationName;

????????????? this.dataGridOrderDetails.DataSource = _ds;

????????????? this.dataGridOrderDetails.DataMember = _ds.Customers.TableName+"."+_ds.Customers.ChildRelations[0].RelationName+"."+_ds.Orders.ChildRelations[0].RelationName;

???????? }

对于上面的三个表的动态关联,你也可以使用SetDataBinding方法来完成数据的动态绑定,而不是分别指定DataGride的DataSource和DataMemger属性。

this.dataGridCustomers.SetDataBinding(_ds,_ds.Customers.TableName);

this.dataGridOrders.SetDataBinding(_ds,_ds.Customers.TableName+"."+_ds.Customers.ChildRelations[0].RelationName);

this.dataGridOrderDetails.SetDataBinding(_ds,_ds.Customers.TableName+"."+_ds.Customers.ChildRelations[0].RelationName+"."+_ds.Orders.ChildRelations[0].RelationName);

}

数据填充事件处理如下:????????????????????????

private void buttonFillData_Click(object sender, System.EventArgs e)

???????? {

????????????? _ds.Clear();//重新填充数据集

????????????? _dataAccess.FillCustomerOrdersInfo(_ds);

????????????? //_dataAccess.FillCustomerOrdersInfoWithSqlHelper(_ds);

???????? }

执行上面的事件处理函数我们会看到数据显示到对应的DataGrid上,如(图2-1)所示。

如果使用数据读取器获取多表纪录下面是实现的一种方式(参考):

SqlCommand comm = new SqlCommand("GetCustomerOrdersInfo",_conn);

comm.CommandType = CommandType.StoredProcedure;

_conn.Open();

SqlDataReader reader = comm.ExecuteReader();

do

{

???? while(reader.Read())

???? {

???????? Console.WriteLine(reader[0].ToString());//获取数据代码

???? }

}while(reader.NextResult());

Console.ReadLine();

_conn.Close();

?

?

时间: 2024-08-31 03:14:59

ADO.NET中的多数据表操作浅析—读取的相关文章

ADO.NET中的多数据表操作浅析—修改

ado|数据 ADO.NET中的多数据表操作浅析-修改 作者:郑佐??????? 2004-8-5 三.更新数据集 首先需要说明的是我这里去掉了Order Details表,对两个表的操作只是其中的几个字段.下面是窗体界面: 图3-1 单选框用来选择不同的更新方法. 在DataAccess类中增加两个类成员变量: ???? private SqlDataAdapter _customerDataAdapter; //客户数据适配器 ???? private SqlDataAdapter _ord

ADO.NET中的多数据表操作浅析之读取

ado|数据 在开发基于.NET平台的数据库应用程序时,我们一般都会用到DataSet,作为ADO.NET的核心类它为我们提供了强大的功能,而整个看上去就像是放在内存内的一个小型数据库,内部包括了DataTable.DataView.DataRow.DataColumn.Constraint以及DataRelation.当初看到它时真的是有点兴奋. 下面根据我的一些经验来举例说明在ADO.NET中的多表填充.关联表更新以及多个Command对象执行过程中启用事务的操作.欢迎大家交流,或在Blog

ADO.NET中的多数据表读取操作

在开发基于.NET平台的数据库应用程序时,我们一般都会用到DataSet,作为ADO.NET的核心类它为我们提供了强大的功能,而整个看上去就像是放在内存内的一个小型数据库,内部包括了DataTable.DataView.DataRow.DataColumn.Constraint以及DataRelation.当初看到它时真的是有点兴奋. 下面根据我的一些经验来举例说明在ADO.NET中的多表填充.关联表更新以及多个Command对象执行过程中启用事务的操作.欢迎大家交流,或在Blog上留言. 一.

初谈ADO.NET中利用DataAdapter进行数据操作

ado|数据     对于刚刚学习ADO.NET数据访问技术的来说,DataAdapter也许往往会令他们感到迷惑,特别是习惯于利用托拽DataAdapter控件进行开发的朋友,通过这一ADO.NET数据访问控件,我们甚至不用书写一行代码,就能完成各种需要的数据访问和操作,然而在享受快捷便利的同时,心中却总挂着一丝意犹未尽的感觉.DataAdapter控件在背后为我们做了怎样的工作?弄清楚这一点,对于喜欢探根究底的朋友,不仅乐哉悠哉,也有利于我们更它的理解ADO.NET的数据访问机制:同时对于.

C#中对注册表的操作

注册表 Windows 操作系统的注册表包含了很多有关计算机运行的配置方式,打开注册表我们可以看到注册表是按类似于目录的树结构组织的,其中第二级目录包含了五个预定义主键分别是:HKEY_CLASSES_ROOT,HKEY_CURRENT_USER,HKEY_LOCAL_MACHINE,HKEY_USERS,HKEY_CURRENT_CONFIG. 下面我们来分别解释这5个类的作用 HKEY_CLASSES_ROOT该主键包含了文件的扩展名和应用程序的关联信息以及Window Shell和OLE用

VC中对注册表进行操作

近来由于需要在自己写的程序中对注册表进行操作.总结些经验,并做个一个DEMO供日后使用,现在把它拿出来和大家分享- 为了使用方便,把一些操作写成了函数,以便方便调用,具体代码如下所示: 一.定义 HKEY hKey; char content[256]; //所查询注册表键值的内容 DWORD dwType=REG_SZ; //定义读取数据类型 DWORD dwLength=256; struct HKEY__*RootKey; //注册表主键名称 TCHAR *SubKey; //欲打开注册表

ASP.NET中实现Form表单字段值自动填充到操作模型中

  这篇文章主要介绍了ASP.NET中实现Form表单字段值自动填充到操作模型中,本文模仿MVC模式中的自动映射表单了模型,使用泛型和反射实现,需要的朋友可以参考下 我们知道ASP.NET MVC有个强大的地方就是Form表单提交到action的时候,可以直接将Form的参数直接装配到action的参数实体对象中 比如 代码如下: action方法 Register(UserModel userModel) { ............................. } 在提交表单的时候,会

MySQL中多表操作和批处理详细介绍

  多表操作 在一个数据库中,可能存在多个表,这些表都是相互关联的.我们继续使用前面的例子.前面建立的表中包含了员工的一些基本信息,如姓名.性别.出生日期.出生地.我们再创建一个表,该表用于描述员工所发表的文章,内容包括作者姓名.文章标题.发表日期. 1.查看第一个表mytable的内容: mysql> select * from mytable; +----------+------+------------+-----------+ | name | sex | birth | birtha

sql中怎样在多表联合查询后的表中进行增删改操作,新手求大神详细讲解

问题描述 sql中怎样在多表联合查询后的表中进行增删改操作,新手求大神详细讲解 我在C#的winform的DataGridView中显示联合查询的结果,下面就是4张表的联合查询的语句,我需要在DataGridView显示的表中做删除操作,只需要知道删除的sql语句就行,用存储过程也好,级联删除也好,希望详细,我已经在这个问题上困扰很久了,万分感谢! 解决方案 在做C#桌面程序时,我们通常会遇到删除dataGridview中选中的一和或都多行数据,在网上查了好多方法,有些只能删除dataGridv