DataSet & DataTable &DataRow 深入浅出
本篇文章适合有一定的基础的人去查看 ,最好学习过一定net 编程基础在来查看此文章。
1.概念
DataSet是ADO.NET的中心概念。可以把DataSet当成内存中的数据库,DataSet是不依赖于数据库的独立数据集合。所谓独立,就是说,即使断开数据链路,或者关闭数据库,DataSet依然是可用的,DataSet在内部是用XML来描述数据的,由于XML是一种与平台无关、与语言无关的数据描述语言,而且可以描述复杂关系的数据,比如父子关系的数据,所以DataSet实际上可以容纳具有复杂关系的数据,而且不再依赖于数据库链路。我们可以把DataSet当成内存数据库,DataSet里面可以存储多个表(DataTable);我们称DataSet为数据集对象。
2.DataSet 的常见使用
在C# 中 DataSet 存在于 System.Data 命名空间下,类似于java中包的路径。DataSet 有两种事列方式,代码如下
using System;
using System.Data; namespace testData
{
class Program
{
static void Main(string[] args)
{
DataSet ds1 = new DataSet();//无构造实例,不指定DataSetName
Console.WriteLine(ds1.DataSetName);
DataSet ds2 = new DataSet("MySet");//一个构造实例,指定DataSetName
Console.WriteLine(ds2.DataSetName);
Console.ReadKey();
}
}
}

DataSet 常用的属性就一个,Tables ,这里就不过多解释了,如下列出DataSet .Tables 如下常用属性:
// ds.Tables.Count;//获取数据集中存在的表的个数
// ds.Tables.Add(new DataTable());//添加一个表(DataTable)到数据集
//DataTable[] array = { new DataTable(), new DataTable(), new DataTable() };
//ds.Tables.AddRange(array);//添加一个表(DataTable)的数组(Array)到数据集
//ds.Tables.Remove();//删除一个表
//ds.Tables.RemoveAt();//按照表的索引从数据集删除一个表
//ds.WriteXml()//将xml 文件写入到DataSet 数据集中
//ds.ReadXml()//将一个xml 文件读取到数据集中
//ds.Tables.CanRemove()//验证是否可以删除一个集合中的对象。
3.DataTable 的常见使用
DataTable 和DataSet 命名空间一样,实例化的方式有三种,但是常用的就两种,第三种指定了表空间;这里不过多解释,现在我们来看下两种常用实例方式;
实例化1 DataTable dt0 = new DataTable();//没有指定表名,默认表名为 NewDataTable
实例化2 DataTable dt1 = new DataTable("User");//指定表名
using System;
using System.Data; namespace testData
{
class Program
{
static void Main(string[] args)
{
DataSet ds = new DataSet("Set");
DataTable dt = new DataTable("User");
ds.Tables.Add(dt);//把一个表加入到数据集中
Console.WriteLine(ds.Tables.Count);
Console.WriteLine(ds.Tables[0].TableName);
Console.Read();
}
}
}
通过上面的代码,我们可以看出,我们使用DataSet 的Add方法将一个DataTable 加入数据集。
在前文概述中,我们就说了DataTable 是一个数据表,数据表就要有表头,那么如果创建一个表头呢,上代码
DataSet ds = new DataSet("Set");
DataTable dt = new DataTable("User");
dt.Columns.Add(new DataColumn("ID", typeof(int)));//设定表头ID
dt.Columns.Add("Name", typeof(string));//设定表头Name
dt.Columns.Add("Age", Type.GetType("System.Int32"));//设定表头Name
for (int i = ; i < dt.Columns.Count; i++)
{
Console.WriteLine("遍历表头方式1_____"+dt.Columns[i]);
}
foreach (var item in dt.Columns)
{
Console.WriteLine("遍历表头方式2~~~" + item);
}
ds.Tables.Add(dt);//把一个表加入到数据集中
Console.WriteLine("当前数据集中有“"+ ds.Tables.Count + "”个表 ");
Console.WriteLine("当前数据集的表名 " + ds.Tables[].TableName);
Console.Read();
这样一张空的表格,我们就有了,下边就是我们如何添加一个数据行了,别急,我们先来看下DataRow的使用。关于创建一个数据行我们会在高级部分分解。
这里简要说明下DataTable 的其他用法
1 DataTable的复制和数组复制是一样的使用,就是完全Copy 一个方法。
2.DataTable 的克隆,克隆指的是克隆表结构,含主键列
3.我们使用Merge 合并连个表
下列出上诉简要代码
static void Main(string[] args)
{
DataTable dt = new DataTable();
dt.Columns.AddRange(
new DataColumn[]
{
new DataColumn("ID",typeof(int)),
new DataColumn("UserName",typeof(string)),
new DataColumn("Age",typeof(int))
}
);
dt.Rows.Add(, "xiaoli", );
dt.PrimaryKey = new DataColumn[] { dt.Columns[] };//给哪一列设置主键
DataTable dtCopy = dt.Copy();//复制表,连数据和表结构一起复制
DataTable dtClone = dt.Clone();//克隆一个表的结构,不包含数据
DataRow dr= dtClone.NewRow();
dr[] = ;
dr[] = "xiaomei";
dr[] = ;
dtClone.Rows.Add(dr);
dt.Merge(dtClone);//当主键不同的时候,就会出现合并到一起,如果相同以被合并的表为主,表结构不一致也可以合并,这里就不掩饰了
Console.WriteLine(dt.Rows.Count);
Console.Read();
; }
4.DataRow 的常见使用
DataRow 的创建方式,只有一种,因为我们net 中设定了DataRow不允许实例化,只能通过 DataTable.NewRow()来创建,如 DataRow dr = dt.NewRow();这样我们就得到了一个空行对象。
4.1 如何给DataTable 增加行数据呢??看如下代码
using System;
using System.Data; namespace testData
{
class Program
{
static void Main(string[] args)
{
DataSet ds = new DataSet("Set");
DataTable dt = new DataTable("User");
dt.Columns.Add(new DataColumn("ID", typeof(int)));//设定表头ID
dt.Columns.Add("Name", typeof(string));//设定表头Name
dt.Columns.Add("Age", Type.GetType("System.Int32"));//设定表头Name
for (int i = ; i < dt.Columns.Count; i++)
{
Console.WriteLine("遍历表头方式1_____"+dt.Columns[i]); }
foreach (var item in dt.Columns)
{
Console.WriteLine("遍历表头方式2~~~" + item);
}
DataRow dr = dt.NewRow();
dr["ID"] = ;
dr["Name"] = "xiaomeng";
dr["age"] = ;
dt.Rows.Add(dr);
ds.Tables.Add(dt);//把一个表加入到数据集中
for (int i = ; i < dt.Rows.Count; i++)
{
DataRow item= dt.Rows[i];
Console.WriteLine(item["name"]+"今年"+item["age"]+"岁了");
}
Console.WriteLine("当前数据集中有“"+ ds.Tables.Count + "”个表 ");
Console.WriteLine("当前数据集的表名 " + ds.Tables[].TableName);
Console.Read();
}
}
}
我们很多程序员在开发的时候,都会认为这就把一个数据行添加到DataTable 里面了,其实不是这样的?具体看高级部分
5.综合应用之CRUD(增删改查)
5.1新增
我们在DataRow中是有一个属性叫RowState 叫做行状态,行状态主要有如下值:“UnChange”(无变化),“Added”(新增过后的数据),“Modified”(修改之后的数据),“Deleted” 删除中的数据,如下图,下图为上面添加行数据的代码,断点调试。

通过上诉分析,我们会发现,添加的数据,并没有提交到内存上,只是程序临时存储的。那么我们怎么提交呢,看如下代码。
using System;
using System.Data; namespace testData
{
class Program
{
static void Main(string[] args)
{
DataSet ds = new DataSet("Set");
DataTable dt = new DataTable("User");
dt.Columns.Add(new DataColumn("ID", typeof(int)));//设定表头ID
dt.Columns.Add("Name", typeof(string));//设定表头Name
dt.Columns.Add("Age", Type.GetType("System.Int32"));//设定表头Name
for (int i = 0; i < dt.Columns.Count; i++)
{
Console.WriteLine("遍历表头方式1_____"+dt.Columns[i]); }
foreach (var item in dt.Columns)
{
Console.WriteLine("遍历表头方式2~~~" + item);
}
DataRow dr = dt.NewRow();
dr["ID"] = 1;
dr["Name"] = "xiaomeng";
dr["age"] = 21;
dt.Rows.Add(dr);
dt.AcceptChanges();//提交数据
ds.Tables.Add(dt);//把一个表加入到数据集中
for (int i = 0; i < dt.Rows.Count; i++)
{
DataRow item= dt.Rows[i];
Console.WriteLine(item["name"]+"今年"+item["age"]+"岁了");
}
Console.WriteLine("当前数据集中有“"+ ds.Tables.Count + "”个表 ");
Console.WriteLine("当前数据集的表名 " + ds.Tables[0].TableName);
Console.Read();
}
}
}

提交代码DataTable.AcceptChanges();那么有内存的提交是不是也应当有内存的回滚,这里不过多介绍了,我们来看下
DataTable 新增行数据的时候状态 为 Added ,这个时候可以使用DataTable的 .AcceptChanges()方法进行提交,可以使用RejectChanges()函数进行回滚。
DataTable 修改行数据的时候状态 为 Modified ,这个时候可以使用DataTable的 .AcceptChanges()方法进行提交,可以使用RejectChanges()函数进行回滚。
DataTable 删除行数据的时候状态 为 Deleted,这个时候可以使用DataTable的 .AcceptChanges()方法进行提交,可以使用RejectChanges()函数进行回滚。
下面来看下增删改的代码
using System;
using System.Data; namespace testData
{
class Program
{
static void Main(string[] args)
{
DataTable dt = new DataTable();
DataColumn[] arrayColumn = {
new DataColumn("id", typeof(int)),
new DataColumn("name", typeof(string)),
new DataColumn("age", typeof(int)),
new DataColumn("sex", typeof(string))
};
dt.Columns.AddRange(arrayColumn);
Console.WriteLine("---------------------------新增---------------------------");
DataRow dr0 = null;
for (int i = ; i < ; i++)
{
dr0 = dt.NewRow();
dr0[] = i;
dr0[] = "xiaoming"+i;
dr0[] = +i+new Random().Next(,);
dr0[] = new Random().Next(i, )%==?"男":"女"; dt.Rows.Add(dr0);
}
Console.WriteLine("新增没有提交打印");
Print(dt);
Console.WriteLine("状态==================" + dr0.RowState);
dt.AcceptChanges();//新增提交
Console.WriteLine("新增提交后打印");
Print(dt);
Console.WriteLine("状态==================" + dr0.RowState);
Console.WriteLine("---------------------------修改---------------------------");
DataRow dr1 = dt.Rows[];
dr1["name"] = "wbcsky";
Console.WriteLine("修改没有提交打印");
Print(dt);
Console.WriteLine("状态==================" + dr1.RowState);
dt.AcceptChanges();//新增提交
Console.WriteLine("修改提交后打印");
Print(dt);
Console.WriteLine("状态==================" + dr1.RowState);
Console.WriteLine("---------------------------删除---------------------------");
DataRow dr2=dt.Rows[] ;//删除第一条
dr2.Delete();//这里没有使用dt.rmove 和dt.rmoveat,因为这两个方法直接提交了
Console.WriteLine("删除没有提交打印");
Print(dt);
Console.WriteLine("状态==================" + dr2.RowState);
dt.AcceptChanges();//新增提交
Console.WriteLine("删除提交后打印");
Print(dt);
Console.WriteLine("状态==================" + dr2.RowState);
Console.Read();
} private static void Print(DataTable dt)
{
foreach (DataRow item in dt.Rows)
{
string msg = "";
try
{
msg = $"id={ item[0]},name={item[1]},age={item[2]},sex={item[3]}";
}
catch (Exception e)
{
msg =e.Message; } Console.WriteLine(msg);
}
}
}
}
执行结果如下图

6.综合应用之筛选排序
我们如果想给上面的表排序,要怎么排序呢?????,其实排序我们使用DataTable.Select 方法就可以,我们看下怎么使用

我们会看到select 方法有四个重载,我们只用一个参数的和连个参数的,其中一个参数的是赛选,两个参数的第一个参数为筛选,第二个为排序
using System;
using System.Data; namespace testData
{
class Program
{
static void Main(string[] args)
{
DataTable dt = new DataTable();
DataColumn[] arrayColumn = {
new DataColumn("id", typeof(int)),
new DataColumn("name", typeof(string)),
new DataColumn("age", typeof(int)),
new DataColumn("sex", typeof(string))
};
dt.Columns.AddRange(arrayColumn); DataRow dr0 = null;
for (int i = ; i < ; i++)
{
dr0 = dt.NewRow();
dr0[] = i;
dr0[] = "xiaoming"+i;
dr0[] = +i+new Random().Next(,);
dr0[] = new Random().Next(i, )%==?"男":"女"; dt.Rows.Add(dr0);
}
Console.WriteLine("---------------------------筛选之前---------------------------");
Print(dt);
DataRow[] rowArr= dt.Select("age >25", " age desc");
Console.WriteLine("---------------------------筛选之后按年龄排序---------------------------");
PrintRow(rowArr);
Console.Read();
}
private static void PrintRow(DataRow[] rowArr)
{
foreach (DataRow item in rowArr)
{
string msg = "";
try
{
msg = $"id={ item[0]},name={item[1]},age={item[2]},sex={item[3]}";
}
catch (Exception e)
{
msg = e.Message; } Console.WriteLine(msg);
}
}
private static void Print(DataTable dt)
{
foreach (DataRow item in dt.Rows)
{
string msg = "";
try
{
msg = $"id={ item[0]},name={item[1]},age={item[2]},sex={item[3]}";
}
catch (Exception e)
{
msg =e.Message; } Console.WriteLine(msg);
}
}
}
}
DataSet & DataTable &DataRow 深入浅出的更多相关文章
- DataSet - DataTable - DataRow 读取 xml 文件 + 搜索
DataSet ds = XmlHelper.GetDataSetByXml(AppDomain.CurrentDomain.BaseDirectory + "/Config/ConfigN ...
- DataTable,DataSet,DataRow与DataView
DataTable和DataSet可以看做是数据容器,比如你查询数据库后得到一些结果,可以放到这种容器里,那你可能要问:我不用这种容器,自己读到变量或数组里也一样可以存起来啊,为什么用容器?原因是,这 ...
- DataSet,DataTable,DataColumn,DataRow的常用操作
DataSet 这个玩意没什么好讲的,当ado.net查询出有多张表集合的数据返回时,这个时候就会使用到DataSet. DataTable //表之间直接赋值 dt2=dt1; 两者指向同一内存空间 ...
- dataset datatable datacolums datarow
DataSet 表示数据在内存中的缓存. 属性 Tables 获取包含在 DataSet 中的表的集合. ds.Tables["sjxx"] DataTable 表示内存中数据的 ...
- C#使用DataSet Datatable更新数据库的三种实现方法
本文以实例形式讲述了使用DataSet Datatable更新数据库的三种实现方法,包括CommandBuilder 方法.DataAdapter 更新数据源以及使用sql语句更新.分享给大家供大家参 ...
- 手写DataSet,DataTable
一:DataSet DataSet ds = new DataSet();//创建DataSet DataTable dt = new DataTable();//创建一个DataTalbe dt.C ...
- DataSet DataTable操作
DataSet ds = new DataSet(); DataTable dt = new DataTable("OrderList"); ...
- 使用DataSet Datatable 更新数据库的三种方式
1:自动生成命令的条件 CommandBuilder 方法a)动态指定 SelectCommand 属性b)利用 CommandBuilder 对象自动生成 DataAdapter 的 DeleteC ...
- DataSet,DataTable排序(转载)
DataSet,DataTable排序 关于对已经绑定的DataSet的排序的问题: DataSet ds=new DataSet();DataView dv=new DataView();dv. ...
随机推荐
- AngularJs创建一个带参数的自定义方法
学习这篇之前,先要从这篇<AngularJs创建自定义Service>http://www.cnblogs.com/insus/p/6773894.html 开始. 看看: app.con ...
- Google guava cache源码解析1--构建缓存器(1)
此文已由作者赵计刚授权网易云社区发布. 欢迎访问网易云社区,了解更多网易技术产品运营经验. 1.guava cache 当下最常用最简单的本地缓存 线程安全的本地缓存 类似于ConcurrentHas ...
- Android sharedUserId 和系统权限
sharedUserId 给不同的应用使用同一个 sharedUserId 可以运行在这几个应用间互相访问数据(数据库,SharedPreferences,文件). sharedUserId 一旦使用 ...
- Spring学习笔记1——IOC: 尽量使用注解以及java代码
在实战中学习Spring,本系列的最终目的是完成一个实现用户注册登录功能的项目. 预想的基本流程如下: 1.用户网站注册,填写用户名.密码.email.手机号信息,后台存入数据库后返回ok.(学习IO ...
- C#6.0语言规范(十六) 异常
C#中的异常提供了一种结构化,统一且类型安全的方法来处理系统级和应用程序级错误条件.C#中的异常机制与C ++的异常机制非常相似,但有一些重要的区别: 在C#中,所有异常必须由派生自的类类型的实例表示 ...
- python -猜字小游戏
代码运行效果如下: 注意: 1.必须要在python3环境想使用 2.QQ:3084276329(一起交流学习) 3.还请大家评论 Guess the word game代码如下: #! /usr/b ...
- Maven - Maven速成
Maven Maven是一个项目构建和管理工具,有助于开发者快速完成项目的配置,快速建立开发环境,从而提高开发效率. 管理项目构建(build)的生命周期(清理.编译.测试.打包.发布.部署.报告等) ...
- .gitignore总结
git进行管理时,.gitignore是必不可少的,可以指定不需要提交到仓库的资源.最好在git init之后就创建 .gitignore文件,这是个好习惯,常用的配置及说明如下:
- Redis 入门知识
Redis 的前世今生 Redis的诞生于2008年,由Salvatore Sanfilippo开发.最初作者在开发一个网站时,需要实现一个高性能的队列功能,在使用Mysql无果后,决定自己造一个轮子 ...
- 剑指offer七之斐波那契数列
一.题目 大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项.n<=39. 二.思路 序号: 0 1 2 3 4 5 ...