Bootstrap

cast函数_SQL干货|为你打开一扇窗—窗口函数

前言

很早之前就想写一篇关于窗口函数的文章,因精力有限所以一直搁置了(一脸认真的自我检讨),在这篇文章的准备阶段,我也拜读了一些相关文章,总体来说基本上所涉及的窗口函数相关知识均有涉及,但一万个读者有一万个哈姆雷特,每个人的文章角度不同,本文主要是面向基础,希望能够用通俗易懂的话语同时结合实际高频面试题为大家打开一扇“窗”。

正文

一、窗口函数简介

窗口函数也称为OLAP函数。OLAP是online analytical processing的简称,意思是对数据库数据进行实时分析处理。——百度百科

窗口函数的基本语法如下:

<窗口函数> over (partition by <用于分组的列名> order by <用于排序的列名>)

窗口函数是针对定义的行集合执行聚集分析,不减少原表的行数同时具备分组与排序的功能;通俗一点讲,窗口函数会按照指定的列名分组,在组内进行排序,同时保持原表行数不发生变化;这与group by 分组聚合后每组返回一个值(原表行数改变)有明显的区别。

窗口函数关键字:

over() : over子句中的内容为窗口函数的作用域(范围)

partition by :分组

order by : 排序

二、常用窗口函数介绍

1、排序函数

  • rank() 如果有并列名次的行,会占用下一名次的位置
  • dense_rank() 如果有并列名次的行,不会占用下一名次的位置
  • row_number() 不考虑并列名次的情况
select *, rank() over (order by 金额 asc) as ranking, 
dense_rank() over (order by 金额 asc) as dens_rank, 
row_number() over (order by 金额 asc) as row_num from table

7bd1be7bbfa557f556d5a4dce8230bb7.png
排序函数区别示意表

常用场景:

排序函数常用于组内排序问题,如:topN问题 (找出每个部门排名前N的员工)

2、偏移函数

--基本语法

lag(exp_str,offset,defval) over()

Lead(exp_str,offset,defval) over()

--exp_str要取的列

--offset取偏移后的第几行数据

--defval:没有符合条件的默认值
  • lag() lag是用于统计窗口内往上(向前偏移)第n行值
  • lead() lead是用于统计窗口内往下(向后偏移)第n行值
select *,
    lag(要偏移的列) over(partition by department order by 要排序的列) as next_lag
from table
---------------------------------------------------------------------------------------
select *,
    lead(要偏移的列) over(partition by department order by 要排序的列) as next_lead
from table

980e4f5717f821aaabbf04c7f58eb308.png
偏移函数lag示意图

98c7de1a3a2c0e9a7343aa5bfb9088d6.png
偏移函数lead示意图

常用场景:

多用于解决用户连续登录问题,如连续7天都登陆的用户数

拓展:

偏移函数还有first_value()、last_value(),感兴趣的同学可以自行学习,本文不作介绍。

3、聚合函数

  • sum()
  • avg()
  • count()
  • max()
  • min()
select * ,sum(需要计算的列名) over (order by 需要计算的列名) as sum,
avg(需要计算的列名) over (order by 需要计算的列名) as avg,
count(需要计算的列名) over (order by 需要计算的列名) as cot,
max(需要计算的列名) over (order by 需要计算的列名) as max,
min(需要计算的列名) over (order by 需要计算的列名) as min
from table

聚合函数在窗口函数中是针对自身记录、以及自身记录之上的所有数据进行计算。

聚合函数作为窗口函数,可以直观的看到,截止到本行数据,统计数据是多少(最大值、最小值等);同时可以看出每一行数据,对整体统计数据的影响。

三、代码实操(经典问题)

本文代码只展示基本框架,如遇同类问题,直接代入使用即可

1、排序

topN问题 (找出每个部门排名前N的员工)

select *
from (
   select *, 
          row_number() over (partition by 要分组的列名
                       order by 要排序的列名 desc) as ranking
   from 表名) as a
where ranking <= N

2、偏移

user_name 用户名(连续7天都登陆的用户数)

思路:

  1. 首先利用偏移窗口函数lead()求得每个用户在每个登陆时间向后偏移7行的登陆时间;(需要注意:date 按倒序排序)
  2. 再计算每个用户在每个登陆时间滞后7天的登陆时间;
  3. 如果每个用户向后偏移7行的登陆时间等于滞后7天的时间,说明该用户连续登陆了7天
select b.user_name
from
(select user_name,
date,lead(date,7,-1) 
over(partition by user_name order by date desc) as date_7
from user_login_table) b 
where 
date_sub(cast(b.date as date,7)) = cast(b.date_7 as date)

四、总结

本文主要介绍了窗口函数的基本概念以及几种高频窗口函数的用法,在实际工作中窗口函数是经常用到的,可能乍一看不是很好理解,这是正常现象,多练习肯定没问题。

以上,如果觉得有点用,请记得点赞关注,谢谢!

;