目录
- 聚合函数
- 数字函数
- 1. ABS函数:返回一个数的绝对值。
- 2. CEIL函数:返回大于等于给定数的最小整数。
- 3. FLOOR函数:返回小于等于给定数的最大整数。
- 4. ROUND函数:将一个数四舍五入到指定的小数位。
- 5. MOD函数:返回两个数相除的余数。
- 6. POWER函数:返回一个数的指定次幂。
- 7. SQRT函数:返回一个数的平方根。
- 8. EXP函数:返回一个数的指数值。
- 9. LN函数:返回一个数的自然对数。
- 10. LOG函数:返回一个数的对数,可以指定底数。
- 11. TRUNC() 数值截断函数 (不四舍五入)
- 日期函数
- 获取系统当前时间
- 日期加减函数
- 1. 在日期上加上指定的月份。
- 2. 在日期上加上指定的天数。
- 3. 在日期上加上指定的小时数。
- 4. 在日期上加上指定的分钟数。
- 5. 在日期上加上指定的秒数。
- 日期转换函数
- 1. TO_DATE(date_string, format)
- 2. TO_CHAR(date, format)
- 日期间隔函数
- 月份间隔函数
- 日期截断函数
- 返回指定日期当月的最后一天 LAST_DAY(日期)
- 四舍五入式处理日期 ROUND
- 字符串处理函数
- 字符转换函数:TO_CHAR()
- 字符拼接函数
- 列转行函数 wm_concat(x)
- 补充:行转列
- 字符长度 LENGTH()
- 字节长度 LENGTHB()
- 位置查找函数 INSTR(x,str,m,n)
- 替换函数REPLACE(字符串,旧的值,新的值)
- 大小写转换
- 截取函数 LTRIM(x,y)、RTRIM(x,y) 、TRIM(y,x)、SUBSTR(x,m,n)
- 填充函数
- 1. LPAD函数
- 2. RPAD函数
- 3. RTRIM函数
- 4. LTRIM函数
- 5. TRIM函数
- 逻辑函数
- 空值处理 nvl()、nvl2()
- decode()函数
- 可以与case when … 进行替换
- 分析函数
- 聚合函数+开窗函数
- over() 里面不填内容
- over(partition by 字段)分组
- over( order by 字段) 排序
- over(partition by 分组 order by 排序)
- 分析函数配合 rows between and 使用
- 排名函数
- row_number()over()
- rank()over()
- dense_rank()over()
- 有partition by 分组,就在组内进行排名
- 偏移函数
- lag()over() 向下偏移
- lead()over() 向上偏移
- 通过偏移计算 环比情况
聚合函数
- 聚合函数:同时对多行进行操作,并返回一个结果 (别名 多行函数)。
- 聚合函数 结合 GROUP BY 一起使用。
- 使用 GROUP BY 以后 ,聚合函数会根据分组字段,每个组返回一个计算结果。
- 聚合函数有:
- 计算最大值:MAX()
- 计算最小值:MIN()
- 计算总和:SUM()
- 计算平均值:AVG()
- 计算总条数:COUNT()
select max(sal) from emp; --查找员工表中工资的最大值 5000
select deptno,max(sal) from emp group by deptno; --按照部门编号进行分组,找出每个部门中对应的员工工资的最大值。select min(sal) from emp; --查找员工表中工资的最小值 800
select deptno,min(sal) from emp group by deptno; --按照部门编号进行分组,找出每个部门中对应的员工工资的最小值。select sum(sal) from emp; --求所有员工工资的总和 29025
select deptno,sum(sal) from emp group by deptno; --按照部门编号进行分组,统计每个部门里面员工的工资总和。select round(avg(sal),2) from emp; --求所有员工工资的平均值 2073.21
select deptno,round(avg(sal),2) from emp group by deptno; --按照部门编号进行分组,统计每个部门里面员工的工资的平均值。
# 注:round( x ,y )函数为四舍五入函数,指对数据 x 进行四舍五入,保留 y 位小数。select count(empno) from emp; --求员工表中的员工数 14
select deptno,count(empno) from emp group by deptno; --按照部门编号进行分组,统计每个部门里面员工的人数。
数字函数
1. ABS函数:返回一个数的绝对值。
案例:
SELECT ABS(-1) FROM DUAL; -- 1SELECT ABS(-1.544) FROM DUAL; -- 1.544SELECT ABS(0) FROM DUAL; -- 0
2. CEIL函数:返回大于等于给定数的最小整数。
案例:
SELECT CEIL(4.23) FROM DUAL; -- 5SELECT CEIL(-4.23) FROM DUAL; -- -4SELECT CEIL(0) FROM DUAL; -- 0
3. FLOOR函数:返回小于等于给定数的最大整数。
案例:
SELECT FLOOR(4.23) FROM DUAL; -- 4SELECT FLOOR(-4.23) FROM DUAL; -- -5
4. ROUND函数:将一个数四舍五入到指定的小数位。
案例:
ROUND(X,Y) 四舍五入函数 -- 会进行四舍五入SELECT ROUND(123.456,2) FROM DUAL; -- 123.46
5. MOD函数:返回两个数相除的余数。
案例:
SELECT MOD(8,2) FROM DUAL; -- 0SELECT MOD(8,3) FROM DUAL; -- 2SELECT MOD(8,3.2) FROM DUAL; -- 1.6
6. POWER函数:返回一个数的指定次幂。
案例:
SELECT POWER(2,3) FROM DUAL; --2的3次方 8SELECT POWER(2,3.4) FROM DUAL; -- 10.556SELECT POWER(-2,2) FROM DUAL; --4 注:当x是负数时,y必须是整数,不能是小数;
7. SQRT函数:返回一个数的平方根。
案例:
SELECT SQRT(16) FROM dual; -- 4
8. EXP函数:返回一个数的指数值。
案例:
SELECT EXP(2) FROM dual; -- 7.38905609893065
9. LN函数:返回一个数的自然对数。
案例:
SELECT LN(10) FROM dual; -- 2.30258509299405
10. LOG函数:返回一个数的对数,可以指定底数。
案例:
SELECT LOG(100, 10) FROM dual; - 2
11. TRUNC() 数值截断函数 (不四舍五入)
案例:
TRUNC(X,Y) --不会进行四舍五入SELECT TRUNC(123.456,2) FROM DUAL; --在小数点之后第二位截断 123.45SELECT TRUNC(123.456,-1) FROM DUAL; --在小数点之前第一位截断 120SELECT TRUNC(123.456,0) FROM DUAL; -- 123
日期函数
获取系统当前时间
select sysdate from dual; --获取系统时间select current_date from dual; --获取系统时间# 时间格式为:2024/7/5 8:49:32
日期加减函数
1. 在日期上加上指定的月份。
语法:ADD_MONTHS(date, num_months)
案例:
SELECT ADD_MONTHS(sysdate, 6) FROM dual; -- 当前日期加上6个月后的日期。
SELECT ADD_MONTHS(sysdate, -6) FROM dual; -- 当前日期减去6个月后的日期。
2. 在日期上加上指定的天数。
语法:date + num_days
案例:
SELECT sysdate + 7 FROM dual; -- 当前日期加上7天后的日期。
SELECT sysdate - 7 FROM dual; -- 当前日期减去7天后的日期。
3. 在日期上加上指定的小时数。
语法:date + (num_hours / 24)
案例:
SELECT sysdate + (3 / 24) FROM dual; -- 当前日期加上3小时后的日期。
SELECT sysdate - (3 / 24) FROM dual; -- 当前日期减去3小时后的日期。
4. 在日期上加上指定的分钟数。
语法:date + (num_minutes / (24 * 60))
案例:
SELECT sysdate + (30 / (24 * 60)) FROM dual; -- 当前日期加上30分钟后的日期。
SELECT sysdate - (30 / (24 * 60)) FROM dual; -- 当前日期减去30分钟后的日期。
5. 在日期上加上指定的秒数。
语法:date + (num_seconds / (24 * 60 * 60))
案例:
SELECT sysdate + (45 / (24 * 60 * 60)) FROM dual; -- 当前日期加上45秒后的日期。
SELECT sysdate - (45 / (24 * 60 * 60)) FROM dual; -- 当前日期减去45秒后的日期。
日期转换函数
1. TO_DATE(date_string, format)
- 这个函数将一个日期字符串转换为一个日期值。
- date_string 是要转换的日期字符串,而 format 则是日期字符串的格式。例如:
select TO_DATE('01-01-2022', 'DD-MM-YYYY') from dual; -- 返回日期值 2022-01-01。
select TO_DATE('2022-01-01', 'YYYY-MM-DD') from dual; -- 返回日期值 2022-01-01。SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI:SS') AS formatted_date FROM dual; -- 返回当前日期和时间的格式化字符串,例如:'2021-12-21 14:30:45'。
- 可以根据需要使用不同的日期格式化选项。以下是一些常用的日期格式化选项:
- YYYY:四位数的年份
- MM:两位数的月份(从01到12)
- DD:两位数的日期(从01到31)
- HH24:24小时制的小时(从00到23)
- MI:两位数的分钟(从00到59)
- SS:两位数的秒(从00到59)
2. TO_CHAR(date, format)
- 这个函数将一个日期值转换为一个指定格式的字符串。
- date 是要转换的日期值,而 format 则是要转换的日期字符串的格式。例如:
select TO_CHAR(SYSDATE, 'DD-MM-YYYY') from dual; -- 返回字符串 '21-12-2021'。SELECT TO_CHAR(your_date_column, 'YYYY-MM-DD HH24:MI:SS') AS formatted_date FROM your_table; -- 将年月日时分秒转换为字符串,例如:'2021-12-21 14:30:45'
日期间隔函数
select sysdate-to_date('20230101','YYYYMMDD') from dual; --间隔 241.36天select to_date('20230101','YYYYMMDD')-sysdate from dual; -- -241.36-- 另一种形式select sysdate-date'2023-01-01' from dual; --间隔241.36天
- 注1:to_date(‘20230101’,‘YYYYMMDD’) 可以将字符串日期转变为指定格式日期输出,'YYYYMMDD’表示为 年/月/日 输出为2023/1/1 。
- 注2:date’年-月-日’ :可表示一个日期
月份间隔函数
select MONTHS_BETWEEN(sysdate,date'2023-01-01') from dual; -- 7.94747834528076(单位 月)select MONTHS_BETWEEN(TO_DATE('31-12-2021', 'DD-MM-YYYY'), TO_DATE('01-01-2021', 'DD-MM-YYYY')) from dual; -- 返回值 12。
日期截断函数
--- 返回这一年的 1月一号 2023/1/1SELECT TRUNC(SYSDATE,'YYYY') FROM DUAL;---- 返回时间所在月的 一号 2023/8/1SELECT TRUNC(SYSDATE,'MM') FROM DUAL;--- 天,不带时分秒 2023/8/30 8:50:13 2023/8/30SELECT SYSDATE,TRUNC(SYSDATE,'DD') FROM DUAL;--- 当前所在星期的第一天 2023/8/30 8:50:13 2023/8/27(周日) SELECT SYSDATE,TRUNC(SYSDATE,'D') FROM DUAL;--- 返回当前时间所在季度的第一天 2023/8/30 8:50:13 2023/7/1SELECT SYSDATE,TRUNC(SYSDATE,'Q') FROM DUAL;
返回指定日期当月的最后一天 LAST_DAY(日期)
SELECT LAST_DAY(SYSDATE) FROM DUAL; -- 2023/8/31 8:57:36
--TRUNC 函数返回的结果作为 LAST_DAY 函数的参数 ,函数嵌套SELECT LAST_DAY(TRUNC(SYSDATE,'DD'))FROM DUAL; -- 2023/8/31
四舍五入式处理日期 ROUND
---如果为“YEAR”则舍入到某年的1月1日,即前半年舍去,后半年作为下一年SELECT ROUND(SYSDATE,'YEAR') FROM DUAL; --下半年 2024/1/1SELECT ROUND(date'2023-05-23','YEAR') FROM DUAL; --上半年 2023/1/1
---如果为“MONTH”则舍入到某月的1日,即前半月舍去,后半月作为下一月。SELECT ROUND(SYSDATE,'MONTH') FROM DUAL; --2023/9/1SELECT ROUND(SYSDATE-15,'MONTH') FROM DUAL; --2023/8/1
---默认为“DDD”,即月中的某一天,最靠近的天,前半天舍去,后半天作为第二天SELECT ROUND(to_date('20230411092334','YYYYMMDD HH12MISS'),'DDD') FROM DUAL; --2023/4/11SELECT ROUND(SYSDATE,'DDD') FROM DUAL; --2023/8/30
----如果为“DAY”则舍入到最近的周的周日,即上半周舍去,下半周作为下一周周日SELECT ROUND(SYSDATE,'DAY') FROM DUAL; --2023/8/27SELECT ROUND(SYSDATE+3,'DAY') FROM DUAL; --2023/9/3SELECT ROUND(to_date('20230809120000','YYYYMMDD HH12MISS'),'DAY') FROM DUAL; --2023/8/13
字符串处理函数
字符转换函数:TO_CHAR()
select to_char(to_date('20230819','YYYYMMDD'),'Q') from dual; --返回当前时间所在季度 3select to_char(sysdate,'day') from dual; --返回当前日期所在的星期 星期三 SELECT TO_CHAR(SYSDATE,'YYYY') FROM DUAL; --返回当前所在年份 2023SELECT TO_CHAR(SYSDATE,'MM') FROM DUAL; --返回当前所在月份 08SELECT TO_CHAR(SYSDATE,'DD') FROM DUAL; --返回当前所在月的日 30SELECT TO_CHAR(SYSDATE,'YYYYMMDD') FROM DUAL; --返回当前所在日期的年月日 20230830
字符拼接函数
select concat(concat(e.ename,e.empno),'hahaha') from emp e; --例 SMITH7369hahahaselect concat('HELLO','ORACLE') from dual; --HELLOORACLE## 推荐这种写法select 'hello' || ' world !' from dual; -- hello world !
列转行函数 wm_concat(x)
wm_concat(x) 跟group by 一起使用,将每一组的多行的某个字段的数据拼接到一行,并逗号隔开
select e.deptno,wm_concat(e.ename) from emp e group by e.deptno; -- 按照部门进行分组,将同组的员工姓名拼接到同一行。--加distinct 去重select e.deptno,wm_concat(distinct e.job) 拼接 from emp e group by e.deptno; -- 按照部门编号分组,将同部门的工作岗位进行拼接。
补充:行转列
Oracle数据库中并没有内置的行转列函数,但可以使用pivot操作实现行转列的功能。pivot操作允许您将行数据转化为列数据,并将列的值从行数据中聚合。
以下是使用pivot操作实现行转列的示例:
假设有一个名为"sales"的表,记录了每个销售人员在不同月份的销售额:
salesperson month sales_amount
----------------------------------
John Jan 1000
John Feb 1500
John Mar 800
Alice Jan 1200
Alice Feb 1800
Alice Mar 900
现在我们希望将每个销售人员的销售额按月份转化为列数据。可以使用pivot操作实现:
SELECT *
FROM
(SELECT salesperson, month, sales_amountFROM sales
)
PIVOT
(SUM(sales_amount)FOR month IN ('Jan', 'Feb', 'Mar')
)
ORDER BY salesperson;
执行上述查询后,将得到以下结果:
salesperson Jan Feb Mar
-----------------------------
Alice 1200 1800 900
John 1000 1500 800
这样就将原表中的每个月份的销售额转化为了列数据,方便了数据的分析和查看。
字符长度 LENGTH()
select length('ABCF') from dual; -- 4select length('中国') from dual; -- 2
字节长度 LENGTHB()
select lengthb('ABCF') from dual; --4 一个英文占一个字节select lengthb('中国') from dual; --4 一个汉字占两个字节
位置查找函数 INSTR(x,str,m,n)
x表示原字符串,str目标字符,m从哪个位置开始找(默认为 1 ,正数从左往右找),n表示第几次出现(默认为1)。返回值为找到的目标位置,没有找到返回为 0
select instr('hello word','o',1,1) from dual; -- 5select instr('hello word','o',6,2) from dual; -- 0select instr('helloword','o',-1,1) from dual; -- 7select instr('helloword','or') from dual; -- 7
替换函数REPLACE(字符串,旧的值,新的值)
select replace('helloword','or','MM') from dual; --or 替换成 MM ,得到 hellowMMdselect replace('helloword','o','S') from dual; --o 替换成 S ,得到 hellSwSrdselect replace('helloword','ord','D') from dual; --连续字符才可以替换 ord 替换成 D, 得到 hellowDselect replace(ename,'S','MM'),ename from emp;
大小写转换
select upper('helloword') from dual; -- HELLOWORDselect lower('HELLO ORACLE') from dual; -- hello oracle
截取函数 LTRIM(x,y)、RTRIM(x,y) 、TRIM(y,x)、SUBSTR(x,m,n)
- 截去函数截去掉包含在后面的参数 y 中的任意一个或多个字符,要求包含在参数 y 内,并且是连续的。
- LTRIM(x,y) 从左边开始,截去 x 中包含在参数 y 中的字符,默认截去空格
- RTRIM(x,y) 从右边开始,截去 x 中包含在参数 y 中的字符,默认截去空格
- TRIM(y from x) 两边截去 x 中包含的参数 y 中的字符,缺省时去空格 --截去只能填一个字符 y
select ltrim('SASCDFVGS','S') from dual; --ASCDFVGSselect rtrim('SASCDFVGS','S') from dual; --SASCDFVGselect trim(' SASCDFVGS ') from dual; --SASCDFVGS 去除两边空格select trim('S' from 'SASCDFVGS') from dual; --ASCDFVG
- SUBSTR(x,m,n)
- 从原字符串 x 中的第 m 位开始截取,截取 n 个长度。
select substr('DDSIYYDSILOVEU',9,6) from dual; --从第九个位置开始截取,截取六个长度 ILOVEUselect substr('DDSIYYDSILOVEU',-6,6) from dual; --从第九个位置开始截取,截取六个长度 ILOVEUselect substr('DDSIYYDSILOVEU',-6) from dual; --从右往左第六个位置开始截取,默认截取到最后位 ILOVEU
填充函数
1. LPAD函数
- 该函数将指定的字符串用指定的字符填充到指定的长度。语法:
LPAD(string, length, pad_char)
SELECT LPAD('Hello', 10, '*') FROM dual; -- *****Hello
2. RPAD函数
- 该函数与LPAD函数类似,只是将填充字符放在字符串的末尾。语法:
RPAD(string, length, pad_char)
SELECT RPAD('Hello', 10, '*') FROM dual; -- Hello*****
3. RTRIM函数
- 该函数用于移除字符串末尾指定的字符。语法:
RTRIM(string, [trim_char])
SELECT RTRIM('Hello***', '*') FROM dual; -- Hello
4. LTRIM函数
- 该函数用于移除字符串开头指定的字符。语法:
LTRIM(string, [trim_char])
SELECT LTRIM('***Hello', '*') FROM dual; -- Hello
5. TRIM函数
- 该函数用于移除字符串两端指定的字符。语法:
TRIM(trim_char FROM string)
SELECT TRIM('*' FROM '*Hello*') FROM dual; -- Hello
逻辑函数
空值处理 nvl()、nvl2()
- nvl(列名,默认值):对空值进行默认值处理
- nvl2(列名,不为空的处理,空值的处理):对含空值的字段的数据,进行空值和非空值这两个情况进行处理
select job,nvl(comm,0) from emp; --将奖金为空值的数据赋予默认值 0 select job,nvl2(comm,sal+comm,0) from emp; --若奖金为空,则赋予默认值0,否则返回为工资+奖金
decode()函数
decode(列名,判断值1,结果值1,判断值2,结果值2,):可以对某个字段多种情况进行精准判断
select ename,job,decode(job,'MANAGER',sal*1.3,'SALESMAN',sal*1.2,sal*1.1) from emp; --根据不同的工作岗位进行 涨薪select ename,decode(comm,null,0,sal+comm) from emp; --若奖金为空,则赋予默认值0,否则返回为工资+奖金
可以与case when … 进行替换
case … when … 数据判断的逻辑语句
# 语法
casewhen 条件判断1 then 执行结果1when 条件判断2 then 执行结果2...else 执行结果end --可用 as 取个别名# 案例
--查询每个人得到工资等级,2000以下是C,2000-3000是B,3000以上是A
select ename,job,case when sal<2000 then 'C'when sal between 2000 and 3000 then 'B'else 'A'end as 工资等级from emp;
分析函数
分析函数 = 聚合函数+开窗函数over()
- 开窗函数:over()
- 分析函数语法:聚合函数()或者单行函数() + over( partition by 分组 order by 排序)
- 分析函数的运用:对表格的数据进行统一的或者是分组的计算,将计算的结果当成一个新的列,直接拼接在表格的后面。
聚合函数+开窗函数
sum()over(),max()over(),min()over(),count()over,avg()over 对整个表格的数据进行计算。
over() 里面不填内容
select e.*,sum(sal)over() from emp e;
select e.*,max(sal)over() from emp e;
select e.*,min(sal)over() from emp e;
select e.*,count(sal)over() from emp e;
select e.*,avg(sal)over() from emp e;
聚合函数得到的结果,生成新的一列数据字段,在查询结果中展示。
over(partition by 字段)分组
对某个分组内的数据进行统计和计算
--将每个部门的工资总和 拼接在 对应部门 每条数据 后面
select emp.*, sum(sal) over(partition by deptno)
from emp;
over( order by 字段) 排序
在整个表格的范围内,对数据进行 ’累计‘ 的统计。
select e.*,sum(sal)over(order by e.empno asc) from emp e; --按员工编号进行升序排序,并累计统计员工工资 如下图所示
select e.*,max(sal)over(order by e.empno asc) from emp e; --按员工编号进行升序排序,并按排序结果一步一步找对对应的最大工资
select e.*,min(sal)over(order by empno asc) from emp e; --按员工编号进行升序排序,并按排序结果一步一步找对对应的最小工资
select e.*,count(1)over(order by e.empno asc) from emp e; --按员工编号进行升序排序,并按排序结果累计统计当前员工人数
select e.*,count(sal)over(order by e.empno asc) from emp e; --按员工编号进行升序排序,并按排序结果累计统计当前员工人数
select e.*,avg(sal)over(order by e.empno asc) from emp e; --按员工编号进行升序排序,并按排序结果计算当前排序情况的平均工资
over(partition by 分组 order by 排序)
在整个表格的范围内,对数据进行某个字段排序的 ’累计‘ 的统计。
select emp.*,
sum(sal)over(partition by deptno order by empno asc)
from emp;
按照部门编号进行了分组,并按照员工编号进行了升序排序。最后逐步统计部门的员工工资的总和。
分析函数配合 rows between and 使用
- CURRENT ROW:当前行
- n PRECEDING:往前n行
- n FOLLOWING:往后n行
- UNBOUNDED PRECEDING:表示从前面的起点
- UNBOUNDED FOLLOWING:表示到后面的终点
SELECT DEPTNO,ENAME,HIREDATE,SAL,SUM(SAL) OVER() AS S1, --所有行相加SUM(SAL) OVER(PARTITION BY DEPTNO) AS S2, --按DEPTNO分组,组内数据相加SUM(SAL) OVER(PARTITION BY DEPTNO ORDER BY HIREDATE) AS S3, --按DEPTNO分组,组内数据累加SUM(SAL) OVER(PARTITION BY DEPTNO ORDER BY HIREDATE ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS S4, --和S3效果相同,--由起点到当前行聚合SUM(SAL) OVER(PARTITION BY DEPTNO ORDER BY HIREDATEROWS BETWEEN 1 PRECEDING AND CURRENT ROW) AS S5, --当前行和前面1行聚合SUM(SAL) OVER(PARTITION BY DEPTNO ORDER BY HIREDATE ROWS BETWEEN 1 PRECEDING AND 1 FOLLOWING) AS S6, --前1行,当前行及后1行聚合SUM(SAL) OVER(PARTITION BY DEPTNO ORDER BY HIREDATE ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) AS S7 --当前行到最后一行聚合FROM EMP;
排名函数
- 排名函数 是一定要排序才能有排名的函数。
- 语法:
单行函数() + over(partition by 分组 order by 排序)
row_number()over()
按照顺序,给每一行定义一个有顺序的排序数字,依次排序。没有并列排名情况 1 2 3 4 5
select emp.*,row_number() over(order by sal desc) r --按工资降序排序。 from emp;
rank()over()
按照顺序,如果有相同的值,那么会产生并列的排名,而且会跳过占用的名次 1 2 3 3 5
select emp.*, rank() over(order by sal desc) rfrom emp;
dense_rank()over()
按照顺序,如果有相同的值,那么会产生并列的排名,但是不会跳过占用的名次 1 2 3 3 4
select emp.*, dense_rank() over(order by sal desc) rfrom emp;
有partition by 分组,就在组内进行排名
select emp.*, row_number() over(partition by deptno order by sal desc) rfrom emp;
偏移函数
- LAG和LEAD函数可以在一次查询中取出同一字段的前N行的数据和后N行的值。
- 语法:
LAG(ARG1,ARG2,ARG3)over(partition by 字段 order by 字段)
- 参数解释:
- LAG 往下面多少行找
- 第一个参数是偏移哪个字段的数据
- 第二个参数是偏移多少行
- 第三个参数是当偏移之后找不到数据,就给的一个默认值
- ORDER BY 必须要有
- PARTITION BY 可以省略,当省略的时候就是整张表的数据做偏移,不分组
- 当 PARTITION BY 存在时,就是在每一组里边进行偏移
lag()over() 向下偏移
SELECT T.*,LAG(T.SAL,2,0)OVER(PARTITION BY T.DEPTNO ORDER BY T.SAL DESC) LAG_SAL
FROM EMP T;
员工按部门进行分组,并按降序进行排序,之后将工资向下偏移 2 个位置,生成新列,没有数据则补充为 0 。
lead()over() 向上偏移
SELECT T.*,LEAD(T.SAL,2,0)OVER(PARTITION BY T.DEPTNO ORDER BY T.SAL DESC) LEAD_SAL
FROM EMP T;
员工按部门进行分组,并按降序进行排序,之后将工资向上偏移 2 个位置,生成新列,没有数据则补充为 0 。
通过偏移计算 环比情况
- 环比:环比的定义和计算方法:环比是指当前时间段的数据与上一个时间段的数据相比较的增长率,通常用百分数表示。环比计算方法如下:环比增长率 = (当前时间段数据 - 上一个时间段数据) / 上一个时间段数据 × 100%
- 数据声明:
--用lag()over()select t.ym,t.amt-lag(t.amt,1)over(order by t.ym) 环比情况FROM SAL t--用lead()over()select t.ym,t.amt-lead(t.amt,1)over(order by t.ym desc) 环比情况FROM SAL torder by t.ym;