那天有个小孩跟我说LINQ(四)转载
1 LINQ TO SQL(代码下载)
我们以一个酒店管理系统的数据库为例子
表结构很简单:GuestInfo(客人信息表),Room(房间表),RoomType(房间类型表),HotelInfo(每个月盈利信息)
整体就是,Room中有个房间类型ID的外键,客人表中有个房间的ID的外键。月盈利信息先不管
准备:新建项目 linq_Ch4控制台程序,新建DB文件夹
右击DB文件夹,添加→新建项
选择LINQ to SQL类,文件名为 Hotel.dbml,添加
打开服务器资源管理器,我的快捷键是Ctrl+W+L
选中这四张表,往已经打开的Hotel.dbml文件中拖放,然后Ctrl+S保存
1.1 查询单表
引入命名空间,让我们可以访问dbml文件
我们有个对象叫 dbml文件名+DataContext组成的一个名称的对象,通过它你就可以操作数据库了
①查出空房间,还可以住两个人以上的房间的信息;查出的结果先按照 最大住人数降序显示结果再按照床位数降序显示结果(锻炼基本单表查询,多条件排序)
代码如下:
HotelDataContext db = new HotelDataContext(); //1. 查出空房间,还可以住两个人以上的房间的信息;查出的结果先按照 最大住人数降序显示结果,再按照床位数降序显示结果 var query1 = from o in db.Room
where o.R_Status == && o.R_MaxPNum >
orderby o.R_MaxPNum descending, o.R_BedNum descending
select o; foreach (var item in query1)
{
Console.WriteLine(string.Format("房间ID:{0}\t{1}\t床位数:{2},最多可住{3}人,描述:{4}", item.R_ID, item.R_Status == ? "空闲" : "有人", item.R_BedNum, item.R_MaxPNum, item.R_Discription));
}
效果图:
1.2 嵌套方式查询
①按房间类型查询出房间的信息
效果图:
代码如下:
HotelDataContext db = new HotelDataContext(); //2.按房间类型查询出房间的信息
var query2 = from b in db.RoomType
select new
{
RoomTypeName = b.RT_Name,
RoomPrice = b.RT_Price,
Items = from c in db.Room
where c.R_RT_ID == b.RT_ID
select c
}; foreach (var item in query2)
{
Console.WriteLine();
Console.WriteLine("房间类型:" + item.RoomTypeName + ",价格是" + Math.Round(item.RoomPrice, ).ToString()); if (item.Items.Count() > )
{
foreach (var r_info in item.Items)
{
Console.WriteLine(string.Format("{0}({1}):{2}", r_info.R_ID, r_info.R_Status == ? "空闲" : "有人", r_info.R_Discription));
}
}
else
{
Console.WriteLine("该类型下没有任何房间!");
}
} Console.ReadKey();
类似的sql语句帮助你理解:
select a.RT_Name,(select top 1 R_ID from dbo.Room b where a.RT_ID=b.R_RT_ID) as RID from dbo.RoomType a
1.3 多表联合查询
①查出客人的信息,房间的类型信息
var query3 = from r in db.Room
join guest in db.GuestInfo on r.R_ID equals guest.G_R_ID
join rt in db.RoomType on r.R_RT_ID equals rt.RT_ID
select new
{
personID = guest.G_ID,
personName = guest.G_Name,
personCardId = guest.G_CardId,
roomTName = rt.RT_Name,
roomStatus = r.R_Status,
personStatus=guest.G_Status,//结账状态
};
foreach (var info in query3)
{
Console.WriteLine(string.Format("{0}:{1},身份证:{2},房间类型名称:{3},结账状态:{5},房间状态:{4}", info.personID, info.personName, info.personCardId, info.roomTName, info.roomStatus == ? "空闲" : "有人", info.personStatus == ? "未结账" : "已结账"));
}
效果图:
1.4 group多个条件分组
①查出客户信息,按照房间号,结账状态分组,统计每种状态的押金数和已收入的钱财数
为了测试真实性,我又添加了一些数据,表中数据如下:
统计查询效果图:
代码如下:
//4.按照房间号,结账状态分组,房间号第一条件,状态第二条件,结算已收入的押金数和收入数信息 var query4 = from a in db.GuestInfo
group a by new { a.G_R_ID, a.G_Status } into g //按照房间号和客人结账状态分组
select new
{
房间号 = g.Key.G_R_ID,
房间结账状态 = g.Key.G_Status == ? "未结账" : "已结账",
押金数 = g.Sum(i => i.G_Deposit),
收入数 = g.Sum(i => i.G_Pay),
统计数=g.Count()
};
foreach (var item in query4)
{
Console.WriteLine(string.Format("房间号(统计了{4}间):{0}\t结账状态:{1}\t押金数:{2}\t收入数:{3}", item.房间号, item.房间结账状态, item.押金数, item.收入数, item.统计数));
}
说明:
LINQ查询表达式的语法既不能接受group子句中的,也不允许一个查询中包含多个group子句
下面两种写法是错误的:
①
②
使用LINQ to SQL进行数据分组查询时效率会有一定的损失。因为LINQ语句最终会被解释为SQL语句执行,但是LINQ语句解释为SQL时会有一定的效率损失,可以使用视图等数据库对象构建高效的数据查询。
1.5 条件方式查询
①用Where
代码如下:
//5.条件查询
var query5_1 = db.GuestInfo.Where(p => p.G_Name == "周恩来");
foreach (var item in query5_1)
{
Console.WriteLine(item.G_Name+"在"+item.G_RentDate.ToString("yyyy年MM月dd日 hh点mm分")+"入住我旅馆");
}
②Enumerable类的Where方法如下所示:
public static IEnumerable<TSource> Where<TSource>(this IEnumerable<TSource> source,Func<TSource,bool> predicate)
其中predicate,用于测试每个元素是否满足条件的一个函数。Where操作符接受的参数类型为Func<TSource,bool>。该类型是具有一个参数并返回布尔值的委托。
接下来,我们来改写一下Where,自定义一个查询方法,叫ConditionQuery<TSource>
public static IEnumerable<TSource> ConditionQuery<TSource>(IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
return source.Where(predicate);
}
使用如下:
//自定义查询,替代where var query5_2 = ConditionQuery<GuestInfo>(db.GuestInfo,item=>item.G_Name=="周恩来");
Console.WriteLine("第二种方式");
foreach (var item in query5_2)
{
Console.WriteLine(item.G_Name + "在" + item.G_RentDate.ToString("yyyy年MM月dd日 hh点mm分") + "入住我旅馆");
}
效果图:
说明:
LINQ to SQL中使用where自居筛选数据时,效率较高。基本上所有LINQ to SQL查询中含有的where自居都会被解释为SQL语句中的where子句部分。
1.6 类似于数据库中的Not Exists的语句
查出一次都没有租出去的房子的信息,contains方法加上!号,效果等同于not exists
//6. 查出一次都没有租出去的房子的信息
var query6 = from r in db.Room
where !(from g in db.GuestInfo
select g.G_R_ID
).Contains(r.R_ID)
select new
{
房间编号 = r.R_ID,
房间说明 = r.R_Discription,
床位数 = r.R_BedNum,
最多可住人数 = r.R_MaxPNum
};
foreach (var item in query6)
{
Console.WriteLine(string.Format("房间编号:{0};床位数:{1};最多可住人数:{2};房间说明:{3}",item.房间编号,item.床位数,item.最多可住人数,item.房间说明));
}
效果图如下:
1.7 分组联接 GroupJoin
表A.GroupJoin(表B,表A外键,表B外键,处理方法)
表A中数据全显示,表B关联上的就显示,否则为空,类似于左查询
代码如下:
//7.分组联接 GroupJoin
var query7 = db.Room.GroupJoin(
db.GuestInfo,
m => m.R_ID,
n => n.G_R_ID,
(m, n) => new
{
房间ID=m.R_ID,
房间状态 = m.R_Status == ? "空闲" : "有人",
已经为您盈利=n.Sum(k=>k.G_Pay)
}
); foreach (var item in query7)
{
Console.WriteLine(string.Format("房间编号:{0};\t{1}\t已经为您盈利{2:C}", item.房间ID, item.房间状态, item.已经为您盈利));
}
效果图:
1.8 聚合操作系列(常用于分组后处理,对比用法可参考1.4例子)
①Count操作:统计次数,例子很简单不写了
②LongCount操作:跟Count一样,就是返回值是Int64类型的
③Sum操作:统计总额,累计
④Max操作:取一组值中的最大值
⑤Min操作:取一组值中的最小值
⑥Average操作:取一组值中的平均值
⑦Aggregate操作:用于对集合中的元素进行自定义的聚合计算
例如:统计今年的总盈利额,我们操作HotelInfo表
效果图:
代码如下:
//8. Aggregate操作例子
decimal[] amountArray = db.HotelInfo.Select(item => item.earnMoney).ToArray<decimal>();
double amountSum=Convert.ToDouble(amountArray.Aggregate((a, b) =>a+b));
Console.WriteLine("一共" + amountSum+"元");
等同于:
decimal amountSum = db.HotelInfo.Sum(item=>item.earnMoney);
1.9 使用视图查询
准备工作:打开Hotel.dbml文件,打开服务器资源管理器,将视图文件夹的视图拖放到dbml文件中
V_SelectRoom只是查询Room和RoomType的一个视图,我们操作的时候把它当做一张表去处理就行了。
代码如下:
var query9 = from vi in db.V_SelectRoom
join g in db.GuestInfo
on vi.R_ID equals g.G_R_ID
where g.G_Status==
select new
{
房间ID = vi.R_ID,
客人姓名 = g.G_Name
}; foreach (var item in query9)
{
Console.WriteLine(item.房间ID+" "+item.客人姓名);
}
效果图:
主要说明的意思是:
使用视图筛选数据,可以极大的提高LINQ to SQL执行查询效率。使用LINQ to SQL查询多表数据时,会带来一定的效率确实,所以先把复杂数据筛选编译为视图,使用LINQ to SQL直接查询视图数据可显著提高数据查询效率。
2.1 使用存储过程查询
准备工作:打开Hotel.dbml文件,打开服务器资源管理器,将存储过程文件夹下的sp_select_roomByRT拖放到dbml文件中,该存储过程是根据房间类型ID获得房间信息
拖放后,在dbml文件右侧
ALTER proc [dbo].[sp_select_roomByRT]
@RT int
as
select R_BedNum,R_ID,R_MaxPNum,R_Status=case when R_Status=1 then '有人' else '空闲' end,R_Discription,R_Count from Room where R_RT_ID=@RT
使用如下,也非常简单:
//10.使用存储过程查询
var query10 = db.sp_select_roomByRT();
foreach (var item in query10)
{
Console.WriteLine(string.Format("房间编号:{0},房间状态:{1},房间描述:{2}", item.R_ID,item.R_Status,item.R_Discription));
}
效果图:
主要说明的意思是:
进行复杂的数据统计、查询、多表联合查询时,可以使用存储过程封装复杂查询来简化LINQ to SQL类的操作,从而提高数据操作效率。当使用DataContext对象的存储过程生成的方法查询数据时,所有数据都会被存储至 ISingleResult<T>泛型集合中。
2.2 使用存储过程操作
准备工作:打开Hotel.dbml文件,打开服务器资源管理器,将存储过程文件夹下的sp_Insert_Room拖放到dbml文件中,该存储过程是创建一个新房间
拖放后,在dbml文件右侧
ALTER proc sp_Insert_Room
@R_RT_ID int,
@R_Discription varchar(),
@R_bedNum int,
@R_MaxPNum int
as
begin
insert Room(R_RT_ID,R_Discription,R_BedNum,R_MaxPNum) values(@R_RT_ID,@R_Discription,@R_bedNum,@R_MaxPNum)
end
使用如下,也非常简单:
db.sp_Insert_Room(, "这是一间于2013年4月7日0:33:43添加进去的新房间,床有9个,最多可住15个人", , );
Console.WriteLine("添加成功!");
效果图:
接下来我们将sp_Insert_RoomReturnInsertID存储过程也拖放到dbml文件中去
存储过程代码:
ALTER proc [dbo].[sp_Insert_RoomReturnInsertID]
@R_RT_ID int,
@R_Discription varchar(),
@R_bedNum int,
@R_MaxPNum int
as
begin
insert Room(R_RT_ID,R_Discription,R_BedNum,R_MaxPNum) values(@R_RT_ID,@R_Discription,@R_bedNum,@R_MaxPNum)
select @@identity;
end
代码如下:
var queryId= db.sp_Insert_RoomReturnInsertID(, "这是一间于2013年4月7日0:33:43添加进去的新房间,床有9个,最多可住15个人", , );
foreach (var item in queryId)
{
Console.WriteLine("添加成功!插入成功后的主键ID为"+item.Column1);
}
效果图:
同理关于delete,update相关的存储过程操作,我就不写了。
那天有个小孩跟我说LINQ(四)转载的更多相关文章
- 那天有个小孩跟我说LINQ(五)转载
2 LINQ TO SQL(代码下载) 我们以一个简单的销售的业务数据库为例子 表结构很简单:Users(购买者(用户)表),Products(产品信息表),Sales(销 ...
- 那天有个小孩跟我说LINQ(七)转载
1 LINQ TO XML(代码下载) 准备:新建项目 linq_Ch7控制台程序,新建一个XML文件夹,我们就轻松地学习一下吧 XDocument ...
- 那天有个小孩跟我说LINQ(六)转载
2 LINQ TO SQL完结(代码下载) 我们还是接着上次那个简单的销售的业务数据库为例子,打开上次那个例子linq_Ch5 2.1 当数据库中的表建立了主外键 ①根据主键获取子表信息 ...
- 那天有个小孩跟我说LINQ(三)转载
1 LINQ TO Objects续2(代码下载) 新建项目 linq_Ch3控制台程序 1.1 操作字符串 ①查找字符串中包含的大写字母,字符串是由多个char类型组 ...
- 那天有个小孩跟我说LINQ(二)转载
1 LINQ TO Objects续(代码下载) 新建项目 linq_Ch2控制台程序,新建一个Entity文件夹 1.1 学生成绩查询(练习Join) 有三张表如下 ...
- 那天有个小孩跟我说LINQ(一) 转载
1 LINQ准备(代码下载) 新建项目 linq_Ch1控制台程序,新建一个Entity文件夹 1.1 对象初始化器 在Entity新建一个类Student,代码如下 using S ...
- 那天有个小孩跟我说LINQ(八)学会Func
文章已经迁移到:http://www.ayjs.net/2013/08/68/ 文章已经迁移到:http://www.ayjs.net/2013/08/68/ 文章已经迁移到:http://www.a ...
- 那天有个小孩教我WCF[一][2/3]
接着上次的继续讲吧 我们开始吧 9.创建数据库 use master go --创建库 if exists(select * from sysdatabases where name='NewsDB' ...
- 那天有个小孩教我WCF[一][1/3]
那天有个小孩教我WCF[一][1/3] 既然是小孩系列,当然要有一点基础才能快速掌握,归纳,总结的一个系列,哈哈 前言: 第一篇嘛,不细讲,步步教你创建一个简单SOA案例,对WCF有个基本的认识,我不 ...
随机推荐
- 通过Java SE 7自带的监控服务(WatchService API)实现类似.NET FileWatcher的功能
Java SE 7 Tutorial中增加了一个监控目录变更情况的示例,用于介绍其新发布的WatchService API. 但对于用惯了.NET FileWatcher的用户而言,如果用于项目我认为 ...
- POJ2492 A Bug's Life 带权并查集
分析:所谓带权并查集,就是比朴素的并查集多了一个数组,记录一些东西,例如到根的距离,或者和根的关系等 这个题,权数组为relation 代表的关系 1 和父节点不同性别,0,和父节点同性别 并查集一 ...
- 基于React服务器端渲染的博客系统
系统目录及源码由此进入 目录 1. 开发前准备 1.1 技术选型1.2 整体设计1.3 构建开发 2. 技术点 2.1 react2.2 redux, react-router2.3 server-r ...
- Java笔记(二十九)……网络编程
概述 网络模型 网络通讯的要素 ip地址:网络中设备的标识符 端口:用于标识同一台设备上不同的进程,有效端口:0~65535,其中0~1024是系统使用端口或者保留端口 TCP与UDP UDP特点: ...
- winpcap使用之捕获数据包
第一种方法,调用回调函数 #include "pcap.h" /* packet handler 函数原型 */ void packet_handler(u_char *param ...
- Linux经久不衰的应用程序
Linux里面的应用程序一贯以高安全性,高性价比(功能/所占空间),此次记录一下Linux里面比较常用的而且经久不衰的应用程序. Shell: bash(它结合了 csh ...
- Robot Framework自动化测试(二)第一个用例
RIDE启动界面: 首先创建一个Test project File-New Project ,选择Directory类型 在创建的文件夹上右键,创建一个Test Suite Openbaidu, NE ...
- Qt 第一步,环境搭建与测试
晚上离散数学课下课后就开始安装Qt. 最先安装的是5.2版本的,这个最新的版本集成了所有必须的工具,不需要配置,直接就可以运行的. 但是,看书和社区教程以及参考资料时候,发现大多是4.8版本的.于是, ...
- 编程实例--for循环,找出0~100之间与8有关的正整数
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 一段经典的node.js 数据库高并发实现
var proxy = new EventProxy();var status = "ready";var select = function(callback){ proxy.o ...