C# 中奇妙的函数–6. 五个序列聚合运算(Sum, Average, Min, Max,Aggregate)
今天,我们将着眼于五个用于序列的聚合运算。很多时候当我们在对序列进行操作时,我们想要做基于这些序列执行某种汇总然后,计算结果。
Enumerable 静态类的LINQ扩展方法可以做到这一点 。就像之前大多数的LINQ扩展方法一样,这些是基于IEnumerable <TSource>序列的操作。
SUM() - 计算整个序列的总和
它有两种形式:
- SUM( )
- 计算整个序列的总值。
- 源类型必须是以下类型之一: int,long,double,decimal,single 或这些类型的可空变种(int?, long?, double?…) 。
- Sum(Func<TSource, X> projection)
- 计算序列投影值的和。
- 从MSDN上,我们得知X必须是以下类型之一:int,long,double,decimal,single 或这些类型的可空变种(int?, long?, double?…) 。
在这里请注意几件事情。
首先,尽管在C#中支持许多类型,SUM()方法-非投影式-只支持int,long,double,decimal,single 。
1: // 正确
2: double[] data = { 3.14, 2.72, 1.99, 2.32 };
3: var result = data.Sum();
4:
5: //不支持
6: short[] shortData = { 1, 2, 5, 7 };
7:
8: // 出现编辑错误
9: var shortResult = shortData.Sum();
还要注意的是,你可以操作上面这些类型允许Null 的可空值变种。在之前我们讨论过,可为空的类型可以是一个棘手的事情,但用SUM()时我们不用担心,因为所有的空值在求和时都排除了 :
var data = new List<int?> { 1, 3, 9, 13, null, 7, 12, null };
var result = data.Sum();
第二,投影形式是一个比较有趣和有用的方法:
为了说明这一点,让我们假设一个简单的POCO Employee:
public sealed class Employee
{
public string Name { get; set; }
public double Salary { get; set; }
public short Dependents { get; set; }
}
var employees = new List<Employee>
{
new Employee { Name = "Bob", Salary = 35000.00, Dependents = 0 },
new Employee { Name = "Sherry", Salary = 75250.00, Dependents = 1 },
new Employee { Name = "Kathy", Salary = 32000.50, Dependents = 0 },
new Employee { Name = "Joe", Salary = 17500.00, Dependents = 2 },
};
然后我们就可以使用投影方式获得Salary 的总值:
var totalSalary = employees.Sum(e => e.Salary);
虽然投影形式表面上似乎被限制在了上述的类型里(int,long,single,double,decimal),但是如果我们使用lambda表达式或匿名表达,投影的形式将允许较短的类型:
employees.Sum(e => e.Dependents);
employees.Sum(delegate(Employee e) { return e.Dependents; });
这是因为lambda表达式和匿名委托的结果可以自动扩大小数值类型(如 short)到 int。
Average() - 返回序列的平均值
Average()方法,就像SUM()一样,只不过它是用总和除以实际涉及到的项目数。涉及到的是什么意思?请记住,SUM( )不包括空值 。Average()是将所有非null值求平均。例如:
var intList = new int?[] { 10, 20, 30, null };
// 返回 20
Console.WriteLine(intList.Average());
MIN() - 返回最小的序列值
MIN()扩展方法用于研究序列,并返回从它的最小值 :
- Min()
- 查找序列中最小的值。
- 抛出异常,如果没有实现IComparable或IComparable<T>。
- 抛出异常,如果序列是空的,源类型是值类型。
- 如果序列是空的,X是引用类型或者Nullable的值类型,则返回 null 。
- Min(Func<TSource, X> projection)
- 返回泛型序列中的最小值
- 如果 TSource 类型实现 IComparable<T>,则此方法使用该实现比较值。 否则,如果 TSource 类型实现 IComparable,则使用该实现比较值。
- 抛出异常,如果序列是空的,X是值类型。
- 如果 TSource 为引用类型,且源序列为空或仅包含为 null 的值,则此函数返回 null。
MIN()支持几乎任何类型,只要该类型实现IComparable或IComparable <T>。因此,它是不限制的数值类型,可以用于任何比较的对象(包括像值类型的DateTime,TimeSpan):
var shortList = new short[] { 1, 3, 7, 9, -9, 33 };
// 返回 -9
var smallest = shortList.Min();
// 根据家庭成员数量找到最小值
var minDependents = employees.Min(e => e.Dependents);
此外,MIN()不使用泛型约束限制那些支持IComparable 接口的类型参数。相反,它抛出一个运行异常来回应如果序列非空,没有在它的对象实现IComparable的接口。
因此如果使用我们的之前定义的Employee类,下面的第一次调用将返回Null(序列为空),第二次调用会抛出(非空,但不包含IComparable的对象序列) 。
var result1 = Enumerable.Empty<Employee>().Min(); var result2 = employees.Min();
var result3 = Enumerable.Empty<int>().Min(); var result4 = Enumerable.Empty<Employee>().Min(e => e.Dependents);
MAX() - 返回最大的序列值
MAX()MIN()的行为完全一样,只不过它返回最大值,而不是最小值。因此,我们可以使用这些序列中的最大值,或从一个序列的预测最大值:
///返回33
VAR biggestShort = shortList.Max(); //返回75250.0
VAR highestSalary = employees.Max(E => e.Salary);
其他方面,请参考Min()。
Aggregate() - 序列的自定义累加器
有三种形式的Aggregate():
- Aggregate(Func<TSource, TSource, TSource> function)
- 适用于一个函数,它接受一个累加器值和下一个值,并返回结果。
- 值和序列类型是相同的。
- 种子值是序列中的第一个值。
- Aggregate(TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> function)
- 应用序列中的一个函数,累加器值和下一个项目,并返回一个结果。
- 值和序列类型可以不同或相同。
- 必须提供一个种子值来初始化,将指定的种子值用作累加器初始值。
- Aggregate(TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> function, Func<TAccumulate, TResult> resultProjection)
- 同上述,将指定的种子值用作累加器的初始值,并使用指定的函数选择结果值。
这可能看起来相当复杂。只要记住 “此方法的工作原理是对 source 中的每个元素调用一次 func。 每次调用 func 时,都将传递序列中的元素和聚合值(作为 func 的第一个参数)。 并用 func 的结果替换以前的聚合值。”
例如,如果我们想要做一个序列中的所有数字的乘法:
var numbers = new int[] { 1, 3, 9, 2 };
// 使用当前总值乘以下一个数值得到新的总值
var product = numbers.Aggregate((total, next) => total * next);
最后的值是: 1 X 3 X 9 X 2 = 54。
下面看看怎么用更复杂的聚合计算, 可能我们想得到这样一个结果 -- 用每个雇员的工资除以家庭总人口数(包括他自己),再将这些数相加:
var weirdCalculation = employees.Aggregate(0.0, (result, next) => result + next.Salary / (next.Dependents + 1));
参照上面的Empolyee 定义,得到的结果是 110458.8333, 为方便理解请看下面的Excel 表格:
所以你看,我们可以做相当复杂的聚合计算,关键是要记住,你所提供的函数留给下一个“元素”,并把它应用到正在运行的“总值”。
摘要
四个简单的和一个可能有点复杂的,这一组功能相当强大!这些方法可以很容易地对序列进行聚合,使你不需要进行循环和自己计算。他们很运行快也很容易使用,他们很容易阅读,他们也被全面测试过了。敬请享受!
C# 中奇妙的函数–6. 五个序列聚合运算(Sum, Average, Min, Max,Aggregate)的更多相关文章
- Hive函数:SUM,AVG,MIN,MAX
转自:http://lxw1234.com/archives/2015/04/176.htm,Hive分析窗口函数(一) SUM,AVG,MIN,MAX 之前看到大数据田地有关于max()over(p ...
- C# 中奇妙的函数–8. String Remove() 和 Replace()
http://www.cnblogs.com/multiplesoftware/archive/2011/09/27/2192710.html 当对字符串进行操作时,我们经常要删除或者是替换一部分子字 ...
- C# 中奇妙的函数–7. String Split 和 Join
很多时候处理字符串数据,比如从文件中读取或者存入 - 我们可能需要加入分隔符(如CSV文件中的逗号),或使用一个分隔符来合并字符串序列. 很多人都知道使用split()的方法,但使用与其对应的Join ...
- C# 中奇妙的函数–String Split 和 Join
很多时候处理字符串数据,比如从文件中读取或者存入 - 我们可能需要加入分隔符(如CSV文件中的逗号),或使用一个分隔符来合并字符串序列. 很多人都知道使用split()的方法,但使用与其对应的Join ...
- 006——php字符串中的处理函数(五)
<?php /** * 一.addslashes() 在预定义字符串前添加反斜杠 * * stripslashes() 把转义字符串前的反斜杠删除 * get_magic_quotes_gpc( ...
- 数组中的reduce 函数理解
第一次见到reduce 是在js 的高级程序设计中,它的意思是把一个数组减少为一个数,举的例子是数组中元素的求和.它接受一个函数作为参数,函数又有两个参数,一个是prev, 前一个值,一个是next, ...
- SQL之开窗函数详解--可代替聚合函数使用
在没学习开窗函数之前,我们都知道,用了分组之后,查询字段就只能是分组字段和聚合的字段,这带来了极大的不方便,有时我们查询时需要分组,又需要查询不分组的字段,每次都要又到子查询,这样显得sql语句复杂难 ...
- (转)笔记320 SQLSERVER中的加密函数 2013-7-11
1 --SQLSERVER中的加密函数 2013-7-11 2 ENCRYPTBYASYMKEY() --非对称密钥 3 ENCRYPTBYCERT() --证书加密 4 ENCRYPTBYKEY() ...
- JavaScript中的普通函数与构造函数比较
问题 什么是构造函数?构造函数与普通函数区别是什么?用new关键字的时候到底做了什么?构造函数有返回值怎么办?构造函数能当普通函数调用吗? thisthis永远指向当前正在被执行的函数或方法的owne ...
随机推荐
- 搭建本地yum源和局域网yum源
搭建本地yum源和局域网yum源 由于很多客户环境是专网,不允许连网,无法使用网上的各种yum源,来回拷贝rpm包安装麻烦,还得解决依赖问题.所以想着搭建个本地/局域网YUM源,方便安装软件. 1 ...
- 1-5-vim编辑器的使用
第1章 vim主要模式介绍,vim命令模式. 1.1 确保系统已经安装了VIM工具 [root@panda ~]# rpm -qf `which vim` [root@panda ~]# rpm ...
- 去掉 Windows 中控件的虚线框(当当 element == QStyle::PE_FrameFocusRect 时,直接返回,不绘制虚线框)
在 Windows 中,控件得到焦点的时候,会显示一个虚线框,很多时候觉得不好看,通过自定义 QProxyStyle 就可以把这个虚线框去掉. 1 2 3 4 5 6 7 8 9 10 11 12 1 ...
- 简单说说Delphi中线程的释放
线程的释放方式有两种:一种是线程在运行完成后自动释放,一种是手动释放. 无论是那种释放,都应该在线程停止后进行释放. 然而线程的停止也有两种情况:一种是不需要设置标志位,直接完成:一种是由于execu ...
- python socket 客服端服务端编程
客服端编程 import socket try: s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) except socket.error a ...
- 采用WebService客户端调用WSDL/SOAP网络报错的解决办法
WebService接口是网络传输控制的重要途径,在Windows系统下运行客户端时,平时一直能正确运行,但某天可能突然会发生调用wsdl soap邮件标头无法识别等莫名其妙的错误提示,出现这种情况一 ...
- windows下捕获dump之Google breakpad_client
breakpad是Google开源的一套跨平台工具,用于dump的处理.很全的一套东西,我这里只简单涉及breakpad客户端,不涉及纯文本符号生成,不涉及dump解析. 一.使用 最简单的是使用进程 ...
- vs2017 cordova调试ios app
https://docs.microsoft.com/en-us/visualstudio/cross-platform/tools-for-cordova/first-steps/ios-guide ...
- delphi 程序强制结束自身(两种方法都暴力)
procedure KillSelf;begin Sleep(1000); if not TerminateProcess(GetCurrentProcessId, 0) then WinExe ...
- 使用scratchbox2建立交叉编译环境
使用scratchbox2建立交叉编译环境,使交叉编译不再烦人..... os:ubuntu 12.04.4 x64 1. 安装相关工具sudo apt-get install debootstrap ...
