ADO.Net——DataSet的应用

本文详细介绍了ADO.Net中的DataSet,它作为内存中的数据库,支持多表、表间关系和数据约束。DataSet不包含数据库连接,数据可以来源于文件。重点讲解了DataSet的特点,如数据的离线性、可随时读写;常用属性如CaseSensitive、DataSetName等;常用方法如AcceptChanges、Clear、Merge等;以及相关事件。还提到了如何创建和管理DataTable、DataRelation,并展示了代码示例和运行实例。
摘要由CSDN通过智能技术生成

一、知识点描述

  • DataSet是表和列结构在内存中的表示方式,DataSet支持多表、表间关系、数据约束等,和关系数据库的模型基本一致。本质上是微型的数据库。包含一组DataTable对象和DataTable之间的连接关系。不包含数据库连接的概念,也就是说其中的数据可以不是来自数据库,可以是读取的文件中的数据。DataTable包含一些DataRow和DataColumn表示数据库表中的行和列,通过他们可以获取表、行和列的元素,且支持主键和外键。

  • DataSet的数据集是与数据库断开的,DataSet中可包括多个DataTable(可将多个查询结构存到一个DataSet中)。而DataTable中又包括多个DataRow、DataColumn,可通过DataRow、DataColumn来查看、操作其中的数据。如果要将操作后的结果集返回给数据库,则用DataAdapter的Update方法。

  • DataSet 可将数据和架构作为 XML 文档进行读写。可使用 WriteXmlSchema 方法将架构保存为 XML架构,并且可以使用 WriteXml 方法保存架构和数据。若要读取既包含架构也包含数据的 XML 文档,请使用 ReadXml 方法。

1、DataSet的特点

(1)DataSet中的数据集是脱机的。

(2)DataSet中的数据集可以在任何时候去读取或修改。

(3)DataSet中中的 DataTable 对象是按条件区分大小写的。(例如,如果一个 DataTable 被命名为“mydatatable”,另一个被命名为“Mydatatable”,则用于搜索其中一个表的字符串被认为是区分大小写的。但是,如果“mydatatable”存在而“Mydatatable”不存在,则认为该搜索字符串不区分大小写)。

(4)典型的多层实现中,创建和刷新 DataSet 并依次更新原始数据的步骤:

A:通过 DataAdapter 使用数据源中的数据生成和填充 DataSet 中的每个 DataTable。
B:通过添加、更新或删除 DataRow 对象更改单个 DataTable 对象中的数据。
C:调用 GetChanges 方法以创建只反映对数据进行的更改的第二个 DataSet。
D:调用 DataAdapter 的 Update 方法,并将第二个 DataSet 作为参数传递。
E:调用 Merge 方法将第二个 DataSet 中的更改合并到第一个中。
F:针对 DataSet 调用 AcceptChanges。或者,调用 RejectChanges 以取消更改。

2、常用属性
CaseSensitive:用于控制DataTable中的字符串比较是否区分大小写。

DataSetName:当前DataSet的名称。如果不指定,则该属性值设置为"NewDataSet"。如果将DataSet内容写入XML文件,DataSetName是XML文件的根节点名称。

DesignMode:如果在设计时使用组件中的DataSet,DesignMode返回True,否则返回False。

HasErrors:表示DataSet中的DataRow对象是否包含错误。如果将一批更改提交给数据库并将DataAdapter对象的ContinueUpdateOnError属性设置为True,则在提交更改后必须检查DataSet的HasErrors属性,以确定是否有更新失败。

NameSpace和Prefix:指定XML命名空间和前缀

Relations:返回一个DataRelationCollection对象。

Tables:获取包含在 DataSet 中的表的集合。通过索引访问DataTable有更好的性能。

DefaultViewManager:获取 DataSet 所包含的数据的自定义视图,以允许使用自定义的 DataViewManager 进行筛选、搜索和导航。

Events:获取附加到该组件的事件处理程序的列表。

Namespace:获取或设置 DataSet 的命名空间。

Prefix:获取或设置一个 XML 前缀,该前缀是 DataSet 的命名空间的别名。

Relations:获取用于将表链接起来并允许从父表浏览到子表的关系的集合。

3、常用方法
AcceptChanges():提交自加载此 DataSet 或上次调用 AcceptChanges 以来对其进行的所有更改。

BeginInit():开始初始化在窗体上使用或由另一个组件使用的 DataSet。初始化发生在运行时。

Clear():通过移除所有表中的所有行来清除任何数据的 DataSet。

Clone():复制 DataSet 的结构,包括所有 DataTable 架构、关系和约束。不要复制任何数据。

Copy():复制该 DataSet 的结构和数据。

CreateDataReader(): 为每个 DataTable 返回带有一个结果集的 DataTableReader,顺序与 Tables 集合中表的显示顺序相同。

EndInit():结束在窗体上使用或由另一个组件使用的 DataSet 的初始化。初始化发生在运行时。

Equals():确定指定的 Object 是否等于当前的 Object。

GetChanges(): 获取 DataSet 的副本,该副本包含自上次加载以来或自调用 AcceptChanges 以来对该数据集进行的所有更改。

GetType():获取当前实例的 Type。

GetXml:返回存储在 DataSet 中的数据的 XML 表示形式。

GetXmlSchema():返回存储在 DataSet 中的数据的 XML 表示形式的 XML 架构。

HasChanges():该值指示 DataSet 是否有更改,包括新增行、已删除的行或已修改的行。

InferXmlSchema():将 XML 架构应用于 DataSet。

Load(): 通过所提供的 IDataReader,用某个数据源的值填充 DataSet。

Merge():将指定的 DataSet、DataTable 或 DataRow 对象的数组合并到当前的 DataSet 或DataTable 中。

OnRemoveRelation():当从 DataTable 中移除 DataRelation 对象时发生。

OnRemoveTable():当从 DataSet 中移除 DataTable 时发生。

ReadXml(): 将 XML 架构和数据读入 DataSet。

ReadXmlSchema(): 将 XML 架构读入 DataSet。

RejectChanges():回滚自创建 DataSet 以来或上次调用 DataSet.AcceptChanges 以来对其进行的所有更改。

Reset():将 DataSet 重置为其初始状态。子类应重写 Reset,以便将 DataSet 还原到其原始状态。

WriteXml(): 从 DataSet 写 XML 数据,还可以选择写架构。

WriteXmlSchema(): 写 XML 架构形式的 DataSet 结构。

4、常用事件

Disposed:添加事件处理程序以侦听组件上的 Disposed 事件。

Initialized:初始化 DataSet 后发生。

MergeFailed:当目标和源 DataRow 的主键值相同且 EnforceConstraints 设置为真时发生。

二、思维导图
思维导图
三、代码片段
1、数据集创建以及数据表关系建立

  sqlCommand.CommandText =                               //指定SQL命令的命令文本;
               "SELECT * FROM tb_Warehouse;"                   //该命令分别查询所有仓库、药品剂型、药品货柜,查询结果将返回多张表;
               + "SELECT * FROM tb_DrugType;"
               + "SELECT * FROM tb_DrugContainer;";
            SqlDataAdapter sqlDataAdapter = new SqlDataAdapter();             //声明并实例化SQL数据适配器,同时借助构造函数,将其SelectCommand属性设为先前创建的SQL命令;
            sqlDataAdapter.SelectCommand = sqlCommand;                 //将SQL数据适配器的查询命令属性指向SQL命令;
            DataSet dataSet = new DataSet();              //声明并实例化数据集,用于保存查得的多张表;
            sqlConnection.Open();                          //打开SQL连接;
            sqlDataAdapter.Fill(dataSet);                 //SQL数据适配器读取数据,并填充数据集;
            sqlConnection.Close();                     //关闭SQL连接;
            DataTable warehouseTable = dataSet.Tables[0];               //声明仓库数据表,对应数据集的表集合中的第1张数据表;
            DataTable drugtypeTable = dataSet.Tables[1];         //声明药品剂型数据表,对应数据集的表集合中的第2张数据表;
            DataTable drugcontainerTable = dataSet.Tables[2];         //声明药品货柜数据表,对应数据集的表集合中的第3张数据表;
            DataRelation[] dataRelations =            //声明数据关系数组;
            {
                new DataRelation                                   //实例化数据关系,实现仓库表、药品剂型表之间的层次关系;
                    ("Warehouse_DrugType"                  //数据关系名称;
                    , warehouseTable.Columns["No"]             //父表的被参照列为仓库表的编号列;
                    , drugtypeTable.Columns["WarehouseNo"]            //子表的参照列为药品剂型表的院系编号列;
                    , false)                                    //不创建约束(父列上的唯一约束、子列上的外键约束);
                , new DataRelation                         //实例化数据关系,实现药品剂型表、药品货柜表之间的层次关系;
                    ("DrugType_DrugContainer"                   //数据关系名称;
                    , drugtypeTable.Columns["No"]                       //父表的被参照列为药品剂型表的编号列;
                    , drugcontainerTable.Columns["DrugTypeNo"]        //子表的参照列为药品货柜表的专业编号列;
                    , false)                                //不创建约束(父列上的唯一约束、子列上的外键约束);
            };
            dataSet.Relations.AddRange(dataRelations);            //将数据关系数组批量加入数据集的关系集合中;
            this.trv_WarehouseUnit.Nodes.Clear();          //树形视图的节点集合清空;
            foreach (DataRow warehouseRow in warehouseTable.Rows)      //遍历仓库数据表中的每一数据行;
            {
                TreeNode warehouseNode = new TreeNode();         //声明并实例化仓库节点,该节点对应当前某个仓库;
                warehouseNode.Text = warehouseRow["Name"].ToString(); //仓库节点的文本设为当前仓库的名称;
                this.trv_WarehouseUnit.Nodes.Add(warehouseNode);    //将仓库节点加入树形视图的(根)节点集合;
                foreach (DataRow drugtypeRow in warehouseRow.GetChildRows("Warehouse_DrugType"))       //借助先前定义的数据关系,遍历当前仓库所在数据行的子行,即下属所有药品剂型;
                {
                    TreeNode drugtypeNode = new TreeNode();                           //声明并实例化药品剂型节点,该节点对应当前某个剂型;
                    drugtypeNode.Text = drugtypeRow["Name"].ToString();          //药品剂型节点的文本设为当前剂型的名称;
                    warehouseNode.Nodes.Add(drugtypeNode);            //药品剂型节点加入当前仓库节点的节点集合,成为第1级节点之一;
                    foreach (DataRow drugcontainerRow in drugtypeRow.GetChildRows("DrugType_DrugContainer"))        //借助先前定义的数据关系,遍历当前药品剂型所在数据行的子行,即下属所有药品货柜;
                    {
                        TreeNode drugcontainerNode = new TreeNode();                     //声明并实例化药品货柜节点,该节点对应当前某个货柜;
                        drugcontainerNode.Text = drugcontainerRow["Name"].ToString();                                       //药品货柜节点的文本设为当前货柜的名称;
                        drugcontainerNode.Tag =drugcontainerRow["Code"];          //药品货柜节点的标签设为当前货柜的编号;
                        drugtypeNode.Nodes.Add(drugcontainerNode);                      //药品货柜节点加入当前药品剂型节点的节点集合,成为第2级节点之一;
                    }
                }
            }

2、树形视图控件的使用

private void trv_WarehouseUnit_AfterSelect(object sender, TreeViewEventArgs e)
        {
            if (this.trv_WarehouseUnit.SelectedNode.Level == 2)                   //若树形视图的选中节点的级别为3,即选中药品货柜节点;
            {
                int ContainerCode = (int)this.trv_WarehouseUnit.SelectedNode.Tag;                //将树形视图的选中节点的标签转为整型,即可获得事先保存的药品货柜编号;
                SqlConnection sqlConnection = new SqlConnection();                 //声明并实例化SQL连接;
                sqlConnection.ConnectionString =
                ConfigurationManager.ConnectionStrings["Sql"].ConnectionString;                //配置管理器从配置文件读取连接字符串,并将之赋予SQL连接的连接字符串属性;
                SqlCommand sqlCommand = new SqlCommand();                   //声明并实例化SQL命令;
                sqlCommand.Connection = sqlConnection;             //将SQL命令的连接属性指向SQL连接;
                sqlCommand.CommandText = "SELECT Name,Validity,Situation FROM tb_Drug WHERE ContainerCode=@ContainerCode;";       //指定SQL命令的命令文本;该命令查询当前选中药品货柜的所有药品名单,以用作数据网格视图数据源;
                sqlCommand.Parameters.AddWithValue("@ContainerCode", ContainerCode);                                    //向SQL命令的参数集合添加参数的名称、值;
                SqlDataAdapter sqlDataAdapter = new SqlDataAdapter();         //声明并实例化SQL数据适配器,同时借助构造函数,将其SelectCommand属性设为先前创建的SQL命令;
                sqlDataAdapter.SelectCommand = sqlCommand;               //将SQL数据适配器的查询命令属性指向SQL命令;
                DataTable drugTable = new DataTable();                 //声明并实例化数据表,用于保存当前选中药品货柜的所有药品名单,以用作数据网格视图的数据源;
                sqlConnection.Open();                                            //打开SQL连接;
                sqlDataAdapter.Fill(drugTable);                           //SQL数据适配器读取数据,并填充班级数据表;
                sqlConnection.Close();                                    //关闭SQL连接;
                this.dgv_InventorySearch.DataSource = drugTable;         //设置数据网格视图的数据源;
                this.dgv_InventorySearch.Columns["Name"].HeaderText = "药品名称";                                         //将数据网格视图的指定列的表头文本设为中文;
                this.dgv_InventorySearch.Columns["Validity"].HeaderText = "有效期";
                this.dgv_InventorySearch.Columns["Situation"].HeaderText = "库存量";
                this.dgv_InventorySearch.Columns[this.dgv_InventorySearch.Columns.Count - 1].AutoSizeMode =         //数据网格视图的最后一列的自动调整列宽模式设为填充(至数据网格视图右侧边缘);
                    DataGridViewAutoSizeColumnMode.Fill;
            }
        }

四、运行实例(DataSet结合树形图控件)
点击载入前:
在这里插入图片描述
点击载入后:
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值