在没学习开窗函数以前,咱们都知道,用了分组以后,查询字段就只能是分组字段和聚合的字段,这带来了极大的不方便,有时咱们查询时须要分组,又须要查询不分组的字段,每次都要又到子查询,这样显得sql语句复杂难懂,给维护代码的人带来很大的痛苦,然而开窗函数出现了,曙光也来临了。若是要想更具体了解开窗函数,请看书《程序员的SQL金典》,开窗函数在mysql不能使用。mysql
开窗函数与聚合函数同样,都是对行的集合组进行聚合计算。它用于为行定义一个窗口(这里的窗口是指运算将要操做的行的集合),它对一组值进行操做,不须要使用group by语句对数据进行分组,可以在同一行中同时返回基础行的列和聚合列。定义看不懂没关系,会用就行。程序员
举个简单例子 查询每一个工资小于5000的员工信息(姓名,城市 年龄 薪水),而且显示小于5000的员工个数,尝试使用下面语句:sql
SELECT FName, FCITY, FAGE, FSalary, COUNT(FName) FROM T_Person WHERE FSALARY<5000
消息 8120,级别 16,状态 1,第 1 行 选择列表中的列 'T_Person.FName' 无效,由于该列没有包含在聚合函数或 GROUP BY 子句中。
可使用子查询实现,语句:函数
SELECT FName, FCITY, FAGE, FSalary, ( SELECT COUNT(FName) FROM T_Person WHERE FSALARY<5000 ) PersonNum FROM T_Person WHERE FSALARY<5000
结果:学习
使用开窗函数实现,查询结果如出一辙,就不粘贴了:fetch
SELECT FName, FCITY, FAGE, FSalary, COUNT(FName) OVER() as PersonNum FROM T_Person WHERE FSALARY<5000
1.开窗函数格式:函数名(列) OVER(选项)spa
2.聚合开窗函数格式:聚合函数(列) OVER(PARTITION BY 字段)3d
over关键字把聚合函数当成聚合开窗函数而不是聚合函数,SQL标准容许将全部的聚合函数用作聚合开窗函数。OVER关键字后的括号中还常常添加选项用以改变进行聚合运算的窗口范围。若是OVER关键字后的括号为空,则开窗函数会对结果集合的全部行进行聚合运算。code
PARTITION BY来定义行的分区来进行聚合运算,与group by 不一样,partition by 字句建立的分区是独立于结果集的,建立的分区只是用于进行聚合运算,并且不一样的开窗函数所建立的分区不互相影响,例如:查询全部人员的信息,并查询所属城市的人员数以及同年龄的人员数:blog
SELECT FName,FCITY, FAGE, FSalary, COUNT(FName) OVER(PARTITION BY FCITY) CityNum, COUNT(FName) OVER(PARTITION BY FAGE) AgeNum FROM T_Person ORDER by FCITY
查询全部人员的信息,并查询所属城市的人员数,每一个城市的人按照年龄排序语句:
SELECT FName,FCITY, FAGE, FSalary, COUNT(FName) OVER(PARTITION BY FCITY ORDER BY FAGE) CityNum FROM T_Person
3.排序开窗函数格式:排序函数() OVER(ORDER BY 字段)
(1)主要函数有ROW_NUMBER()、RANK()、DENSE_RANK()、NTILE()
ROW_NUMBER() 加行号,通常能够用于分页查询(如今被offset fetch取代 ),对于没有主键列的表加行号做用很明显,删除重复数据等。
按照薪水高低给全部人员排序,一样薪水的排名不同,能够用row_number(),
with a as ( SELECT FName, FSalary, FCity, FAge, ROW_NUMBER() over(ORDER BY FSalary) as RowNum FROM T_Person ) SELECT * FROM a
使用rank()将每一个城市的薪水排行,值同样的同一个排名,出现两个第一名的时候,排在两个第一名后的排名将是第三名
SELECT FName, FSalary, FCity, FAge, RANK() over(PARTITION BY FCITY ORDER BY FSalary) as RankNum FROM T_Person
使用dense_rank()将每一个城市的薪水排行,值同样的同一个排名,出现两个第一名的时候,排在两个第一名后的排名将是第三名