LINQ to DataSet,对离线数据的Linq支持、AsEnumeable()
一、DataTable的扩展方法:
1、DataTable转Linq:AsEnumerable 方法
返回的EnumerableRowCollection<DataRow> 实现了 IEnumerable<T>接口。其中泛型参数T是DataRow。
此对象可用在 LINQ表达式或方法查询中。
语法:
public static EnumerableRowCollection<DataRow> AsEnumerable (this DataTable source);
在以下示例中,一个 DataColumn 数据表ProductName
,提取ProductName
值,然后将值输出。
DataTable table = new DataTable();
table.Columns.Add("ID");
table.Columns.Add("ProductName"); table.Rows.Add("1", "Chai");
table.Rows.Add("2", "Queso Cabrales");
table.Rows.Add("3", "Tofu"); var productNames = from products in table.AsEnumerable()
select products.Field<string>("ProductName");
Console.WriteLine("Product Names: ");
foreach (string productName in productNames)
{
Console.WriteLine(productName);
}
注意:转成EnumerableRowCollection后,需要对两个这样的行集合进行操作,如:Distinct,Union,Except,Intesect,SequenceEqual都需要对数据源的元素进行相等比较,需要使用专门为Linq To Datatable新增加的DataRowComparer作为参数(实现了IEqulityComparer接口),用于比较DataRow的值(而不是引用比较)。否则,根本不能达到预期操作。
IEnumerable<DataRow> distinctTable = table.AsEnumerable().Distinct(DataRowComparer.Default);
2、Linq转DataTable
(1)返回DataTable,其中包含的副本DataRow对象。
public static DataTable CopyToDataTable<T> (this IEnumerable<T> source) where T : DataRow;
以下示例查询 SalesOrderHeader 表的订单后 2001 年 8 月 8 日,并使用CopyToDataTable方法来创建DataTable从该查询。
DataTable orders = ds.Tables["SalesOrderHeader"]; IEnumerable<DataRow> query =
from order in orders.AsEnumerable()
where order.Field<DateTime>("OrderDate") > new DateTime(2001, 8, 1)
select order; // Create a table from the query.
DataTable boundTable = query.CopyToDataTable<DataRow>();
(2)副本表的DataRow到指定的对象DataTable。
public static void CopyToDataTable<T> (this IEnumerable<T> source, DataTable table, LoadOption options) where T : DataRow;
LoadOption:当使用 Load 或 Load 方法时控制数据源中的值如何应用至现有行。
- OverwriteChanges: 传入此行的值将同时写入每列数据的当前值和原始值版本。
- PreserveChanges: 传入此行的值将写入每列数据的原始值版本。 每列数据的当前版本不变化。 这是默认设置。
- Upsert : 传入此行的值将写入每列数据的当前版本。 每列数据的原始版本不变化。
以下示例:如果tableold表设置了主键,则可以合并记录,否则追加。
table.AsEnumerable().CopyToDataTable(tableold,LoadOption.OverwriteChanges);
tips:默认情况下(没对表进行操作),数据行的Original版本是不存在的,视图访问将报错。可以在访问之前使用DataRow.HasVersion来判断,也可以通过调用DataRow.AcceptChanges()方法来建立Original版本,避免异常发生。
二、DataRow中的扩展方法:
(1)获取字段值
Field:提供对 DataRow 中的每个列值的强类型访问。
public static T Field<T> (this DataRow row, string columnName);
举例:
foreach (DataRow row in table.AsEnumerable())
{
row.Field<int>("Id");
row.Field<string>("Name", DataRowVersion.Original);
}
(2)设置字段值
SetField:为 DataRow 中的指定列设置一个新值。
public static void SetField<T> (DataRow row, string columnName, T value);
如果value是null,则SetField方法转换null值设置为DBNull.Value。
举例:
var rows = from s in table.AsEnumerable()
where s.Field<string>("Name") == "c"
select s;
rows.Single<DataRow>().SetField("Name", "cc");
三、Linq To DataTable的常见用法。
1、查询:
DataTable table = new DataTable();
table.Columns.Add("ID", Type.GetType("System.Int32"));
table.Columns.Add("PID", Type.GetType("System.Int32"));
table.Columns.Add("CreateDate", Type.GetType("System.DateTime")); table.Rows.Add(1, 2, "2010-1-1");
table.Rows.Add(2, 3, "2011-1-1");
table.Rows.Add(3, 3, "2012-1-1");
table.Rows.Add(4, 2, null); var rows = from s in table.AsEnumerable()
where s.Field<int>("ID") > 1 && !s.IsNull("CreateDate")
select new
{
ID = s["ID"],
CreateDate = s["CreateDate"]
};
rows.ToList().ForEach(m => Console.WriteLine(m.ID + "-" + m.CreateDate));//2-2011-01-01 0:00:00 3-2012-01-01 0:00:00
2、分组:
var query = from r in table.AsEnumerable()
group r by new
{
PID = r.Field<int>("PID")
} into g
select new
{
Pid = g.Key.PID,
FirstCreateDate = g.First().Field<DateTime>("CreateDate"),
IDS = string.Join(",", g.Select(n => n.Field<int>("ID")).ToArray()),
Count = g.Count(q => q.Field<int>("ID") > 1),
Sum1 = g.Sum(q => q.Field<int>("ID"))
};
query.ToList().ForEach(m => Console.WriteLine(m.Pid + "-" + m.FirstCreateDate + "-" + m.IDS));//2-2010-01-01 0:00:00-1,4 3-2011-01-01 0:00:00-2,3
例2:按PID分组,获取倒数第二条信息
var query1 = from r in table.AsEnumerable()
where new int[] { 3, 4 }.Contains(r.Field<int>("PID"))
orderby r.Field<DateTime>("CreateDate") descending
group r by r.Field<int>("PID") into g
let n = (from d in g.Take(2).Skip(1) select d).FirstOrDefault()
select new
{
PID = g.Key,
ID = n.Field<int>("ID"),
CreateDate = n.Field<DateTime>("CreateDate")
};
query1.ToList().ForEach(m => Console.WriteLine(m.PID + "-" + m.ID + "-" + m.CreateDate));//3-2-2011-01-01 0:00:00
3、Join组合查询:
var result = from t1 in table.AsEnumerable()
join t2 in table.AsEnumerable() on
t1.Field<int>("ID") equals t2.Field<int>("ID")
where t2.Field<DateTime?>("CreateDate") < DateTime.Today || !t2.Field<DateTime?>("CreateDate").HasValue
select new
{
ID = t2.Field<int>("ID"),
CreateDate = t1.Field<DateTime?>("CreateDate")
};
result.ToList().ForEach(m => Console.WriteLine(m.ID + "-" + m.CreateDate));
LINQ to DataSet,对离线数据的Linq支持、AsEnumeable()的更多相关文章
- LINQ系列:LINQ to DataSet的DataTable操作
LINQ to DataSet需要使用System.Core.dll.System.Data.dll和System.Data.DataSetExtensions.dll,在项目中添加引用System. ...
- LINQ(LINQ to DataSet)
http://www.cnblogs.com/SkySoot/archive/2012/08/21/2649471.html DataTable.Select()方法使用和 SQL 相似的过滤语法从 ...
- C#数据操作LINQ
Linq是language integrated query的缩写,即"语言集成查询"之意. Linq主要包含四个组件,Linq to object.Linq to XML.Lin ...
- 泛型 Field 和 SetField 方法 (LINQ to DataSet)
LINQ to DataSet 为 DataRow 类提供用于访问列值的扩展方法:Field 方法和 SetField 方法.这些方法使开发人员能够更轻松地访问列值,特别是 null 值.DataSe ...
- C# LINQ系列:LINQ to DataSet的DataTable操作 及 DataTable与Linq相互转换
LINQ to DataSet需要使用System.Core.dll.System.Data.dll和System.Data.DataSetExtensions.dll,在项目中添加引用System. ...
- EntityFramework数据持久化 Linq介绍
一.LINQ概述与查询语法 二.LINQ方法语法基础(重点) 三.LINQ聚合操作与元素操作(重点) 四.数据类型转换(重点) 一.LINQ概述与查询语法 1.LINQ(Language Integr ...
- EntityFramework数据持久化 Linq语法应用
Linq基础语法 LINQ概述 LINQ(Language Integrated Query,语言集成查询)提供了一种跨数据源和数据格式查询的统一模型. LINQ的组成: LINQ To Object ...
- Code First开发系列之管理数据库创建,填充种子数据以及LINQ操作详解
返回<8天掌握EF的Code First开发>总目录 本篇目录 管理数据库创建 管理数据库连接 管理数据库初始化 填充种子数据 LINQ to Entities详解 什么是LINQ to ...
- 8天掌握EF的Code First开发系列之3 管理数据库创建,填充种子数据以及LINQ操作详解
本文出自8天掌握EF的Code First开发系列,经过自己的实践整理出来. 本篇目录 管理数据库创建 管理数据库连接 管理数据库初始化 填充种子数据 LINQ to Entities详解 什么是LI ...
随机推荐
- 日常工作问题解决:Redhat6.5--解决yum无法正常安装配置问题
1.问题描述 解决RedHat6.5下yum功能不能用问题: 在redhat6.5下使用yum安装时,会提示:This system is not registered to Red Hat Subs ...
- S1. Android 功能大全
[概述] 这篇文章主要描述安卓开发中可能实现的功能点. [准备工作] IDE:Android Studio,简单操作 如何创建一个 Android 项目 Android 项目结构分析 Androidm ...
- Spring Boot系列教程十四:Spring boot同时支持HTTP和HTTPS
自签证书 openssl生成服务端证书,不使用CA证书直接生成 -in server.csr -signkey server.key -out server.crt # 5.server证书转换成ke ...
- SQL概要与表的创建
SQL概要与表的创建 1.表的结构 关系数据库通过类似Excel 工作表那样的.由行和列组成的二维表来管理数据.用来管理数据的二维表在关系数据库中简称为表. 根据SQL 语句的内容返回的数据同 ...
- 【C#】课堂知识点#1
标准数字格式字符串 https://docs.microsoft.com/zh-cn/dotnet/standard/base-types/standard-numeric-format-string ...
- 原生js分页器插件
window.page = function page(ele, para) { this.ele = document.querySelector(ele); this.options = { co ...
- ngix介绍
ngix能做什么? 1 反向代理 2 负载均衡 3 正向代理 4 HTTP服务器(动静分离) 1 反向代理 Reverse Proxy 是指以代理服务器来接受来自internet或者是客户端的连接请 ...
- Admui相关第三方插件
ace 版本:1.2.3au 官网:https://github.com/ajaxorg/ace-builds/ 许可:BSD 依赖:无 DataAPI:data-pulgin="ace&q ...
- qt翻译和国际化的探讨。
这段时间一直都在怼qt的国际化,以前也接触过国际化,但是感觉不是那么的深刻,这次是因为一个项目要做一个国际化的版本,代码里面是不能出现中文的,所以就翻译了一下.qt用的是4.8.6 64位的,IDE( ...
- (四)XML基础(客户端和服务端发送与接收xml数据)
案例: index.jsp <%@ page language="java" import="java.util.*" pageEncoding=&quo ...