在Sql server 2012里面,开窗函数丰富了许多,其中带出了2个新的函数 First_Value 和 Last Value .现在来介绍一下这2个函数的应用场景.

首先分析一下First_Value(),用法是根据Partition By对数据进行分区,如果忽略Partition By ,那么默认整块数据一个区域,然后根据Order By 进行排序,取出第一个值。

;WITH CTE AS(SELECT 1 AS ID ,'2016-06-01' AS DT,'A' AS UName,135 AS TotalAmount UNION ALL

SELECT 2 AS ID ,'2016-06-05' AS DT,'A' AS UName,148 AS TotalAmount UNION ALL

SELECT 3 AS ID ,'2016-06-02' AS DT,'B' AS UName,120 AS TotalAmount UNION ALL

SELECT 4 AS ID ,'2016-06-06' AS DT,'B' AS UName,153 AS TotalAmount UNION ALL

SELECT 5 AS ID ,'2016-06-10' AS DT,'B' AS UName,198 ASTotalAmount

)SELECT *,

FIRST_VALUE(CTE.TotalAmount)OVER (PARTITION BY CTE.UName ORDER BY CTE.ID) ASFirstDeal,

FIRST_VALUE(CTE.DT)OVER (PARTITION BY CTE.UName ORDER BY CTE.ID) ASFirstDateFROM CTE

ID DT UName TotalAmount FirstDeal FirstDate

----------- ---------- ----- ----------- ----------- ----------

1 2016-06-01 A 135 135 2016-06-01

2 2016-06-05 A 148 135 2016-06-01

3 2016-06-02 B 120 120 2016-06-02

4 2016-06-06 B 153 120 2016-06-02

5 2016-06-10 B 198 120 2016-06-02

在这个场景里面,我求出了根据用户名称(UName)来进行分区,根据ID进行一个排序,求出每个用户第一次购买商品的时间以及交易的金额。如果不使用First_Value 我们也可以换另外一种写法

;WITH CTE AS(SELECT 1 AS ID ,'2016-06-01' AS DT,'A' AS UName,135 AS TotalAmount UNION ALL

SELECT 2 AS ID ,'2016-06-05' AS DT,'A' AS UName,148 AS TotalAmount UNION ALL

SELECT 3 AS ID ,'2016-06-02' AS DT,'B' AS UName,120 AS TotalAmount UNION ALL

SELECT 4 AS ID ,'2016-06-06' AS DT,'B' AS UName,153 AS TotalAmount UNION ALL

SELECT 5 AS ID ,'2016-06-10' AS DT,'B' AS UName,198 ASTotalAmount

)SELECT *

FROMCTE aCROSS APPLY(SELECT TOP 1 a.TotalAmount AS FirstDeal,DT AS FirstDate FROM CTE WHERE a.UName = CTE.UName ORDER BY CTE.ID) ASb--或者改写成这种形式

;WITH CTE AS(SELECT 1 AS ID ,'2016-06-01' AS DT,'A' AS UName,135 AS TotalAmount UNION ALL

SELECT 2 AS ID ,'2016-06-05' AS DT,'A' AS UName,148 AS TotalAmount UNION ALL

SELECT 3 AS ID ,'2016-06-02' AS DT,'B' AS UName,120 AS TotalAmount UNION ALL

SELECT 4 AS ID ,'2016-06-06' AS DT,'B' AS UName,153 AS TotalAmount UNION ALL

SELECT 5 AS ID ,'2016-06-10' AS DT,'B' AS UName,198 ASTotalAmount

)SELECT a.*,b.TotalAmount AS FirstDeal,b.DT ASFirstDateFROMCTE aLEFT JOIN CTE b ON a.UName = b.UName AND NOT EXISTS(SELECT * FROM CTE WHERE b.UName = UName AND DT < b.DT)

在这三种写法里面,查询的结果是一致的,燃鹅从查询分析器分析的分析来看,查询性能,使用First_Value 的效率最高,not exists 的效率其次,使用Cross Apply 效率最低。

但是这只是从查询这么少量的测试数据反馈出来的结果,如果具体的场景需要应用,最好是结合实际情况看实际的查询计划来得出最适当的查询效果。

然后说下 Last_Value() 的用法,虽然说First_Value 和 Last_Value 一看就想两兄弟。但是!用起来真不是这样的一回事啊!

如果根据First_Value 的解释,那么Last_Value 就是根据Partition进行分区,根据Order By 进行排序返回最后的一个值。想我是这样想的,但是操作起来就不是这么一回事了。

;WITH CTE AS(SELECT 1 AS ID ,'2016-06-01' AS DT,'A' AS UName,135 AS TotalAmount UNION ALL

SELECT 2 AS ID ,'2016-06-05' AS DT,'A' AS UName,148 AS TotalAmount UNION ALL

SELECT 3 AS ID ,'2016-06-02' AS DT,'B' AS UName,120 AS TotalAmount UNION ALL

SELECT 4 AS ID ,'2016-06-06' AS DT,'B' AS UName,153 AS TotalAmount UNION ALL

SELECT 5 AS ID ,'2016-06-10' AS DT,'B' AS UName,198 ASTotalAmount

)SELECT *,

LAST_VALUE(CTE.TotalAmount)OVER ( PARTITION BY CTE.UName ORDER BY CTE.DT) ASLRFROMCTE

ID DT UName TotalAmount LR----------- ---------- ----- ----------- -----------

1 2016-06-01 A 135 135

2 2016-06-05 A 148 148

3 2016-06-02 B 120 120

4 2016-06-06 B 153 153

5 2016-06-10 B 198 198

咦!?说好的根据 UName 进行分组,然后再DT进行排序区最后一个价格呢??完全不是这样子啊!!

对,这才是Last_Value的用法,实际上。在我测试的版本里面 (2012,2014), 除了根据 Partition By 进行分区,还对Order by 不一样的值产生不一样的取值。

所以,如果你想看到这个效果,我们不妨把测试样例数据修改一下,把其中2个DT改成一样的,如下面效果

;WITH CTE AS(SELECT 1 AS ID ,'2016-06-05' AS DT,'A' AS UName,135 AS TotalAmount UNION ALL

SELECT 2 AS ID ,'2016-06-05' AS DT,'A' AS UName,148 AS TotalAmount UNION ALL

SELECT 3 AS ID ,'2016-06-02' AS DT,'B' AS UName,120 AS TotalAmount UNION ALL

SELECT 4 AS ID ,'2016-06-02' AS DT,'B' AS UName,153 AS TotalAmount UNION ALL

SELECT 5 AS ID ,'2016-06-10' AS DT,'B' AS UName,198 ASTotalAmount

)SELECT *,

LAST_VALUE(CTE.TotalAmount)OVER ( PARTITION BY CTE.UName ORDER BY CTE.DT) ASLRFROMCTE

ID DT UName TotalAmount LR----------- ---------- ----- ----------- -----------

1 2016-06-01 A 135 135

2 2016-06-05 A 148 148

3 2016-06-02 B 120 120

4 2016-06-06 B 153 153

5 2016-06-10 B 198 198

so 现在就看到取值是不一样的,燃鹅,确还是有一个问题,到底哪个才是Last_Value 呢??查询计划说了算~(这个我还真没验证过,请各位大神指导一下)

所以啊,不要看名字就觉得First_Value 和 Last_Value 是亲兄弟啊!!是隔壁老王的啊!!

好,本次分享到这里~

mysql first value_开窗函数 First_Value 和 Last_Value相关推荐

  1. 开窗函数-first_value/last_value

    1.开窗函数是什么? 开窗函数用于为行定义一个窗口(这里的窗口是指运算将要操作的行的集合),它对一组值进行操作,不需要使用GROUP BY子句对数据进行分组,能够在同一行中同时返回基础行的列和聚合列. ...

  2. sql 中的导航函数 FIRST_VALUE, LAST_VALUE

    FIRST_VALUE 以下示例计算各部门的最快时间,并且计算某个部门么每个员工与第一名的时间差 WITH finishers AS(SELECT 'Sophia Liu' as name,TIMES ...

  3. mysql开窗函数有哪些_mysql开窗函数

    开窗函数: 它可以理解为记录集合,开窗函数也就是在满足某种条件的记录集合上执行的特殊函数. 对于每条记录都要在此窗口内执行函数,有的函数随着记录不同,窗口大小都是固定的,这种属于静态窗口: 有的函数则 ...

  4. MySQL:开窗函数

    当查询条件需要用到复杂子查询时,聚合函数操作起来非常麻烦,因此使用开窗函数能够轻松实现. 注意:在Oracle中称为分析函数. 在MySQL中称为开窗函数,使用于MySQL8.0以上版本,sql se ...

  5. mysql 排序开窗函数_MySQL中实现开窗函数

    一.概述 row_number是数据库中常用的一个开窗函数,可以实现数据分区编号的功能,然而MySQL并不支持开窗函数.本文介绍了在MySQL中实现开窗函数的方法. 二.经典开窗函数 首先准备基础数据 ...

  6. MySQL8中的开窗函数

    开窗函数 前言 窗口函数的格式 函数(Function)的类型 开窗函数over() 窗口函数使用 ROW_NUMBER() RANK()与DENSE_RANK() LEAD()与LAG() FIRS ...

  7. 大数据之Hive:Hive 开窗函数(二)

    目录 前言 1.last_value开窗函数 2.lag开窗函数 3.lead开窗函数 4.cume_dist开窗函数 前言 书接上回,上回重点讲了聚合函数之count开窗函数,first_value ...

  8. hive中的开窗函数

    目录 count开窗函数 sum开窗函数 min开窗函数 max开窗函数 avg开窗函数 first_value开窗函数 last_value开窗函数 lag开窗函数.lead开窗函数 cume_di ...

  9. sql开窗函数及编程练习题

    今天做题碰见了个没见过的开窗函数 什么是开窗函数 开窗函数也叫分析函数,有两类:一类是聚合开窗函数,一类是排序开窗函数. 聚合开窗函数就是我们常见的avg().count().sum()等.SQL 标 ...

最新文章

  1. Linux内存管理 (4)分配物理页面
  2. 《最后期限》阅读笔记2
  3. [development][PCRE] old PCRE
  4. 同步数据_体验GoldenGate 18C数据同步到ADW
  5. 浅谈工业无线遥控器的基本定义及原理
  6. HTML5 表单相关
  7. Java垂直镜像,Java OpenCV实现图像镜像翻转效果
  8. Windows 7 的一些使用技巧
  9. 小米8绑定账号和设备验证失败_网赚项目刚需账号答疑大集合!
  10. robotstudio工作站建立
  11. lwj_C#_string类方法 可变字符串
  12. [精简]托福核心词汇60
  13. “百练”成钢:NumPy 100练
  14. 【天光学术】演讲稿:微笑面对生活
  15. 想象力无限,人工智能时代你对AGV有什么期许
  16. python解一元二次方程
  17. 判断邮箱的正则表达式
  18. Simulink电力电子仿真——(一)概述1
  19. Mybatis-plus 代码生成器(新)
  20. spice time step too small的解决办法

热门文章

  1. eplan连接定义点不显示_显示无线桥接成功,但是连接副路由器的设备依然上不去网的问题...
  2. java 90 训练营 二期下载_90天java进阶营二期 主流java技术与热门开源项目视频教程...
  3. centos删除gnome_自定义你的 GNOME 桌面主题 | Linux 中国
  4. centos安装mysql-python_在Centos上安装mysql python
  5. matlab与微分方程
  6. Grafana中文版本
  7. C#:统计字符串中每个字符的个数
  8. 复用:设计模式 反模式 分析模式
  9. IIS 服务器下载apk文件报404错
  10. WCF 部署在IIS上