Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 584|回复: 0

[默认分类] LINQ体验(5)——LINQ to SQL语句之Select/Distinct和Count/Sum/Min/Max/Avg

[复制链接]
  • TA的每日心情
    开心
    2021-12-13 21:45
  • 签到天数: 15 天

    [LV.4]偶尔看看III

    发表于 2018-7-13 12:29:44 | 显示全部楼层 |阅读模式
    上一篇讲述了LINQ,顺便说了一下Where操作,这篇开始我们继续说LINQ to SQL语句,目的让大家从语句的角度了解LINQ,LINQ包括LINQ to Objects、LINQ to DataSets、LINQ to SQL、LINQ to Entities、LINQ to XML,但是相对来说LINQ to SQL在我们程序中使用最多,毕竟所有的数据都要在数据库运行着各种操作。所以先来学习LINQ to SQL,其它的都差不多了,那么就从Select说起吧,这个在编写程序中也最为常用。本篇详细说明一下Select和Count/Sum/Min/Max/Avg。
    Select/Distinct操作符
    适用场景:o(∩_∩)o… 查询呗。
    说明:和SQL命令中的select作用相似但位置不同,查询表达式中的select及所接子句是放在表达式最后并把子句中的变量也就是结果返回回来;延迟。
    Select/Distinct操作包括9种形式,分别为简单用法、匿名类型形式、条件形式、指定类型形式、筛选形式、整形类型形式、嵌套类型形式、本地方法调用形式、Distinct形式。
    1.简单用法:
    这个示例返回仅含客户联系人姓名的序列。
    1. var q =
    2.     from c in db.Customers
    3.     select c.ContactName;
    复制代码
    注意:这个语句只是一个声明或者一个描述,并没有真正把数据取出来,只有当你需要该数据的时候,它才会执行这个语句,这就是延迟加载(deferred loading)。如果,在声明的时候就返回的结果集是对象的集合。你可以使用ToList() 或ToArray()方法把查询结果先进行保存,然后再对这个集合进行查询。当然延迟加载(deferred loading)可以像拼接SQL语句那样拼接查询语法,再执行它。
    2.匿名类型形式:
    说明:匿名类型是C#3.0中新特性。其实质是编译器根据我们自定义自动产生一个匿名的类来帮助我们实现临时变量的储存。匿名类型还依赖于另外一个特性:支持根据property来创建对象。比如,var d = new { Name = "s" };编译器自动产生一个有property叫做Name的匿名类,然后按这个类型分配内存,并初始化对象。但是var d = new {"s"};是编译不通过的。因为,编译器不知道匿名类中的property的名字。例如string c = "d";var d = new { c}; 则是可以通过编译的。编译器会创建一个叫做匿名类带有叫c的property。
    例如下例:new{c,ContactName,c.Phone};ContactName和Phone都是在映射文件中定义与表中字段相对应的property。编译器读取数据并创建对象时,会创建一个匿名类,这个类有两个属性,为ContactName和Phone,然后根据数据初始化对象。另外编译器还可以重命名property的名字。
    1. var q =
    2.     from c in db.Customers
    3.     select new {c.ContactName, c.Phone};
    复制代码
    上面语句描述:使用 SELECT 和匿名类型返回仅含客户联系人姓名和电话号码的序列
    1. var q =
    2.     from e in db.Employees
    3.     select new
    4.     {
    5.         Name = e.FirstName + " " + e.LastName,
    6.         Phone = e.HomePhone
    7.     };
    复制代码
    上面语句描述:使用SELECT和匿名类型返回仅含雇员姓名和电话号码的序列,并将FirstName和LastName字段合并为一个字段“Name”,此外在所得的序列中将HomePhone字段重命名为Phone。
    1. var q =
    2.     from p in db.Products
    3.     select new
    4.     {
    5.         p.ProductID,
    6.         HalfPrice = p.UnitPrice / 2
    7.     };
    复制代码
    上面语句描述:使用SELECT和匿名类型返回所有产品的ID以及HalfPrice(设置为产品单价除以2所得的值)的序列。
    3.条件形式:
    说明:生成SQL语句为:case when condition then else。
    1. var q =
    2.     from p in db.Products
    3.     select new
    4.     {
    5.         p.ProductName,
    6.         Availability =
    7.         p.UnitsInStock - p.UnitsOnOrder < 0 ?
    8.         "Out Of Stock" : "In Stock"
    9.     };
    复制代码
    上面语句描述:使用SELECT和条件语句返回产品名称和产品供货状态的序列。
    4.指定类型形式:
    说明:该形式返回你自定义类型的对象集。
    1. var q =
    2.     from e in db.Employees
    3.     select new Name
    4.     {
    5.         FirstName = e.FirstName,
    6.         LastName = e.LastName
    7.     };
    复制代码
    上面语句描述:使用SELECT和已知类型返回雇员姓名的序列。
    5.筛选形式:
    说明:结合where使用,起到过滤作用。
    1. var q =
    2.     from c in db.Customers
    3.     where c.City == "London"
    4.     select c.ContactName;
    复制代码
    上面语句描述:使用SELECT和WHERE返回仅含伦敦客户联系人姓名的序列。
    6.shaped形式(整形类型):
    说明:其select操作使用了匿名对象,而这个匿名对象中,其属性也是个匿名对象。
    1. var q =
    2.     from c in db.Customers
    3.     select new {
    4.         c.CustomerID,
    5.         CompanyInfo = new {c.CompanyName, c.City, c.Country},
    6.         ContactInfo = new {c.ContactName, c.ContactTitle}
    7.     };
    复制代码
    语句描述:使用SELECT 和匿名类型返回有关客户的数据的整形子集。查询顾客的ID和公司信息(公司名称,城市,国家)以及联系信息(联系人和职位)。
    7.嵌套类型形式:
    说明:返回的对象集中的每个对象DiscountedProducts属性中,又包含一个集合。也就是每个对象也是一个集合类。
    1. var q =
    2.     from o in db.Orders
    3.     select new {
    4.         o.OrderID,
    5.         DiscountedProducts =
    6.             from od in o.OrderDetails
    7.             where od.Discount > 0.0
    8.             select od,
    9.         FreeShippingDiscount = o.Freight
    10.     };
    复制代码
    语句描述:使用嵌套查询返回所有订单及其OrderID 的序列、打折订单中项目的子序列以及免送货所省下的金额。
    8.本地方法调用形式(LocalMethodCall):
    这个例子在查询中调用本地方法PhoneNumberConverter将电话号码转换为国际格式。
    1. var q = from c in db.Customers
    2.          where c.Country == "UK" || c.Country == "USA"
    3.          select new
    4.          {
    5.              c.CustomerID,
    6.              c.CompanyName,
    7.              Phone = c.Phone,
    8.              InternationalPhone =
    9.              PhoneNumberConverter(c.Country, c.Phone)
    10.          };
    复制代码
    PhoneNumberConverter方法如下:
    1. public string PhoneNumberConverter(string Country, string Phone)
    2. {
    3.     Phone = Phone.Replace(" ", "").Replace(")", ")-");
    4.     switch (Country)
    5.     {
    6.         case "USA":
    7.             return "1-" + Phone;
    8.         case "UK":
    9.             return "44-" + Phone;
    10.         default:
    11.             return Phone;
    12.     }
    13. }
    复制代码
    下面也是使用了这个方法将电话号码转换为国际格式并创建XDocument
    1. XDocument doc = new XDocument(
    2.     new XElement("Customers", from c in db.Customers
    3.               where c.Country == "UK" || c.Country == "USA"
    4.               select (new XElement("Customer",
    5.                       new XAttribute("CustomerID", c.CustomerID),
    6.                       new XAttribute("CompanyName", c.CompanyName),
    7.                       new XAttribute("InterationalPhone",
    8.                        PhoneNumberConverter(c.Country, c.Phone))
    9.                      ))));
    复制代码
    9.Distinct形式:
    说明:筛选字段中不相同的值。用于查询不重复的结果集。生成SQL语句为:SELECT DISTINCT [City] FROM [Customers]
    1. var q = (
    2.     from c in db.Customers
    3.     select c.City )
    4.     .Distinct();
    复制代码
    语句描述:查询顾客覆盖的国家。
    Count/Sum/Min/Max/Avg操作符
    适用场景:统计数据吧,比如统计一些数据的个数,求和,最小值,最大值,平均数。
    Count
    说明:返回集合中的元素个数,返回INT类型;不延迟。生成SQL语句为:SELECT COUNT(*) FROM
    1.简单形式:
    得到数据库中客户的数量:
    1. var q = db.Customers.Count();
    复制代码
    2.带条件形式:
    得到数据库中未断货产品的数量:
    1. var q = db.Products.Count(p => !p.Discontinued);
    复制代码
    LongCount
    说明:返回集合中的元素个数,返回LONG类型;不延迟。对于元素个数较多的集合可视情况可以选用LongCount来统计元素个数,它返回long类型,比较精确。生成SQL语句为:SELECT COUNT_BIG(*) FROM
    1. var q = db.Customers.LongCount();
    复制代码
    Sum
    说明:返回集合中数值类型元素之和,集合应为INT类型集合;不延迟。生成SQL语句为:SELECT SUM(…) FROM
    1.简单形式:
    得到所有订单的总运费:
    1. var q = db.Orders.Select(o => o.Freight).Sum();
    复制代码
    2.映射形式:
    得到所有产品的订货总数:
    1. var q = db.Products.Sum(p => p.UnitsOnOrder);
    复制代码
    Min
    说明:返回集合中元素的最小值;不延迟。生成SQL语句为:SELECT MIN(…) FROM
    1.简单形式:
    查找任意产品的最低单价:
    1. var q = db.Products.Select(p => p.UnitPrice).Min();
    复制代码
    2.映射形式:
    查找任意订单的最低运费:
    1. var q = db.Orders.Min(o => o.Freight);
    复制代码
    3.元素:
    查找每个类别中单价最低的产品:
    1. var categories =
    2.     from p in db.Products
    3.     group p by p.CategoryID into g
    4.     select new {
    5.         CategoryID = g.Key,
    6.         CheapestProducts =
    7.             from p2 in g
    8.             where p2.UnitPrice == g.Min(p3 => p3.UnitPrice)
    9.             select p2
    10.     };
    复制代码
    Max
    说明:返回集合中元素的最大值;不延迟。生成SQL语句为:SELECT MAX(…) FROM
    1.简单形式:
    查找任意雇员的最近雇用日期:
    1. var q = db.Employees.Select(e => e.HireDate).Max();
    复制代码
    2.映射形式:
    查找任意产品的最大库存量:
    1. var q = db.Products.Max(p => p.UnitsInStock);
    复制代码
    3.元素:
    查找每个类别中单价最高的产品:
    1. var categories =
    2.     from p in db.Products
    3.     group p by p.CategoryID into g
    4.     select new {
    5.         g.Key,
    6.         MostExpensiveProducts =
    7.             from p2 in g
    8.             where p2.UnitPrice == g.Max(p3 => p3.UnitPrice)
    9.             select p2
    10.     };
    复制代码
    Average
    说明:返回集合中的数值类型元素的平均值。集合应为数字类型集合,其返回值类型为double;不延迟。生成SQL语句为:SELECT AVG(…) FROM
    1.简单形式:
    得到所有订单的平均运费:
    1. var q = db.Orders.Select(o => o.Freight).Average();
    复制代码
    2.映射形式:
    得到所有产品的平均单价:
    1. var q = db.Products.Average(p => p.UnitPrice);
    复制代码
    3.元素:
    查找每个类别中单价高于该类别平均单价的产品:
    1. var categories =
    2.     from p in db.Products
    3.     group p by p.CategoryID into g
    4.     select new {
    5.         g.Key,
    6.         ExpensiveProducts =
    7.             from p2 in g
    8.             where p2.UnitPrice > g.Average(p3 => p3.UnitPrice)
    9.             select p2
    10.     };
    复制代码
    Aggregate
    说明:根据输入的表达式获取聚合值;不延迟。即是说:用一个种子值与当前元素通过指定的函数来进行对比来遍历集合中的元素,符合条件的元素保留下来。如果没有指定种子值的话,种子值默认为集合的第一个元素。
    下面用一个表格总结一下这篇说的LINQ to SQL语句


       
       Where
       过滤;延迟
       
       
       Select
       选择;延迟
       
       
       Distinct
       查询不重复的结果集;延迟
       
       
       Count
       返回集合中的元素个数,返回INT类型;不延迟
       
       
       LongCount
       返回集合中的元素个数,返回LONG类型;不延迟
       
       
       Sum
       返回集合中数值类型元素之和,集合应为INT类型集合;不延迟
       
       
       Min
       返回集合中元素的最小值;不延迟
       
       
       Max
       返回集合中元素的最大值;不延迟
       
       
       Average
       返回集合中的数值类型元素的平均值。集合应为数字类型集合,其返回值类型为double;不延迟
       
       
       Aggregate  
       根据输入的表达式获取聚合值;不延迟
      


    本系列链接:LINQ体验系列文章导航
    LINQ推荐资源
    LINQ专题:http://kb.cnblogs.com/zt/linq/ 关于LINQ方方面面的入门、进阶、深入的文章。
    LINQ小组:http://space.cnblogs.com/group/linq/学习中遇到什么问题或者疑问提问的好地方。
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-4-20 15:40 , Processed in 0.439469 second(s), 50 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表