mysql窗口函数rank() over、dense_rank() over、row_number() over 实现分组排行

在做导出时,遇到一个根据价格最低数统计,所以用到了序号排行,数据库用的8.0.33;

正好mysql 8.0以上版本支持窗口函数

总结一下几种函数:

1、row_number()

row_number()over(partition by字段1 order by 字段2) 的结果是每一行记录生成一个序号,依次排序且排序的序号不会重复

2、rank()

rank()over(partition by字段1 order by 字段2) 的结果会考虑排序字段值相同的情况,若排序字段的值相同则其序号是一样的,后续不同字段值的序号为(前一行序号+N,其中N为前一个字段值重复的行数),比如 1 1 3 4 4 4 7。**

3、dense_rank()

dense_rank()over(partition by字段1 order by 字段2) 的结果也会考虑排序字段值相同的情况,即排序字段的值相同那么他们的序号是一样的,但是与rank()****的区别是后续不同字段值的序号为(前一行序号+1),比如 1 1 2 2 3 4 5

根据价格进行序号排行

#row_number() over()使用

	SELECT
		classify_code as '材料code', price_after_tax as '价格排名列',
		row_number() OVER ( PARTITION BY classify_code ORDER BY classify_code, price_after_tax ) AS '排行序号' 
	FROM
		purchase_project_op_material 
	WHERE
		purchase_project_id = 1878637071744167937 
		AND type = 1 

#执行结果:

#rank() over()使用


	SELECT
		classify_code as '材料code', price_after_tax as '价格排名列',
		rank() OVER ( PARTITION BY classify_code ORDER BY classify_code, price_after_tax ) AS '排行序号' 
	FROM
		purchase_project_op_material 
	WHERE
		purchase_project_id = 1878637071744167937 
		AND type = 1 

#执行结果

#dense_rank() over()使用


	SELECT
		classify_code as '材料code', price_after_tax as '价格排名列',
		dense_rank() OVER ( PARTITION BY classify_code ORDER BY classify_code, price_after_tax ) AS '排行序号' 
	FROM
		purchase_project_op_material 
	WHERE
		purchase_project_id = 1878637071744167937 
		AND type = 1 

#执行结果

最后注意:

over(partition by字段1 order by 字段2)中的partition by 字段1 是可以省略的但是order by 字段2 不可省略

select * from( with vtailratio_tmp as ( select distinct d.*,e.OPERATE_WAY,e.min_tail_fare,e.YEAR_DAY,e.TAIL_COMMISSION_RATIO,e.AREA_FLAG,e.MIN_HOSTING,e.ENABLE_DATE,e.MAX_HOSTING from distributor_fund_protocol d, (select a.fund_code,a.distributor_code, nvl(c.OPERATE_WAY,b.OPERATE_WAY) OPERATE_WAY, 0.00 min_tail_fare, decode(nvl(c.YEAR_DAY,b.YEAR_DAY), '365', '1', '2', '0', '3', '3', '0', '2' ) YEAR_DAY, nvl(c.TAIL_COMMISSION_RATIO,b.TAIL_COMMISSION_RATIO) TAIL_COMMISSION_RATIO, nvl(c.AREA_FLAG,b.AREA_FLAG) AREA_FLAG, nvl(c.MIN_HOSTING,b.MIN_HOSTING) MIN_HOSTING, nvl(c.ENABLE_DATE,b.ENABLE_DATE) ENABLE_DATE, nvl(c.MAX_HOSTING,b.MAX_HOSTING) MAX_HOSTING from (select fund_code,distributor_code from fund_info,distributor_info) a, (select * from TAIL_COMMISSION_FARE_SET_ORI where DISTRIBUTOR_CODE = '***') b, (select * from TAIL_COMMISSION_FARE_SET_ORI where DISTRIBUTOR_CODE <> '***') c where a.fund_code = c.fund_code(+) and a.DISTRIBUTOR_CODE = c.DISTRIBUTOR_CODE(+) and a.fund_code = b.fund_code(+) )e where d.fund_code = e.fund_code and d.distributor_code = e.distributor_code ) select a.distributor_code distributor_code,di.distributor_name,a.fund_code fundcode,fi.fund_name,to_char(replace(sum(BALANCE), ','),'FM999,999,999,999,990.00')balance,to_char(replace(sum(SHARES), ','),'FM999,999,999,999,990.00') shares,to_char(replace(avg(balance), ','),'FM999,999,999,999,990.00') avgbalance,to_char(replace(sum(fee), ','),'FM999,999,999,999,990.00') fee, sum(income) income, avg(shares) avgshares, tfa.min_tail_fare,case when sum(fee)>=tfa.min_tail_fare then sum(fee) else 0 end paid_fee, to_char(nvl(TAIL_COMMISSION_RATIO,0)*100,'fm999999990.0099999')||'%' TAIL_COMMISSION_RATIO from( select distributor_code, fund_code, TRANSACTION_CFM_DATE,sum(balance) balance, sum(shares) shares, sum(fee) fee, sum(TAIL_COMMISSION_RATIO) TAIL_COMMISSION_RATIO,sum(income) income,avg(balance) avgbalance,avg(shares) avgshares from( select distributor_code, fund_code, TRANSACTION_CFM_DATE, share_class, sum(BALANCE) BALANCE, sum(SHARES) SHARES, sum(round(BALANCE /(case when YEAR_DAY>0 then YEAR_DAY else year.YearDays end)* nvl(TAIL_COMMISSION_RATIO, 0), 2)) fee, max(nvl(TAIL_COMMISSION_RATIO, 0)) TAIL_COMMISSION_RATIO, sum(nvl(income, 0)) income, avg(BALANCE) avgbalance, avg(SHARES) avgshares from( SELECT A.distributor_code, A.fund_code, 'A' share_class, A.TRANSACTION_CFM_DATE, decode(b.OPERATE_WAY, '0',a.balance, '1',nvl(A.balance, 0) - nvl(A.REINVEST_BALANCE, 0), '2',nvl(A.ORI_HOLD_BALANCE, 0), '3',nvl(A.ORI_HOLD_BALANCE, 0) - nvl(A.ORI_REINVEST_BALANCE, 0)) BALANCE, decode(b.OPERATE_WAY, '0',a.shares, '1',nvl(A.shares, 0) - nvl(A.REINVEST_SHARE, 0), '2',nvl(A.shares, 0), '3',nvl(A.shares, 0) - nvl(A.REINVEST_SHARE, 0)) SHARES, B.TAIL_COMMISSION_RATIO, A.income, nvl(b.YEAR_DAY,0) YEAR_DAY FROM( select a.distributor_code, a.fund_code, 'A' share_class, a.TRANSACTION_CFM_DATE, a.HOLD_DATE, a.HOLD_RATIO, nvl(a.income, 0) income, nvl(a.REINVEST_SHARE, 0) REINVEST_SHARE, nvl(a.REINVEST_BALANCE, 0) REINVEST_BALANCE, nvl(a.ORI_HOLD_BALANCE, 0) ORI_HOLD_BALANCE, NVL(A.ORI_REINVEST_BALANCE, 0) ORI_REINVEST_BALANCE, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_BALANCE, 0) else nvl(a.HOLD_BALANCE, 0) + nvl(a.income, 0) end balance, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_SHARE, 0) else nvl(a.HOLD_SHARE, 0) + nvl(a.income, 0) end shares from FUND_SALE_STAT a,fund_info b WHERE a.INDIVIDUAL_OR_INSTITUTION = '*' and A.TRANSACTION_CFM_DATE >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.TRANSACTION_CFM_DATE <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} and a.fund_code = b.fund_code ) A, ( select fund_code, distributor_code, AREA_FLAG, MIN_HOSTING, MAX_HOSTING, TAIL_COMMISSION_RATIO, ENABLE_DATE,OPERATE_WAY,YEAR_DAY, last_value(end_date)over(partition by fund_code, distributor_code, ENABLE_DATE order by end_date rows between unbounded preceding and unbounded following) end_date from(select fund_code, distributor_code, AREA_FLAG, MIN_HOSTING, MAX_HOSTING, TAIL_COMMISSION_RATIO, ENABLE_DATE,OPERATE_WAY,YEAR_DAY, nvl(lead(ENABLE_DATE)over(partition by fund_code, distributor_code order by ENABLE_DATE, MIN_HOSTING), '20991231') end_date from vtailratio_tmp where AREA_FLAG in ('0','1') ) ) B where A.fund_code = B.fund_code AND A.distributor_code = B.distributor_code and A.HOLD_DATE >= B.ENABLE_DATE and A.HOLD_DATE < B.end_date AND decode(b.OPERATE_WAY, '0',a.balance, '1',nvl(A.balance, 0) - nvl(A.REINVEST_BALANCE, 0), '2',nvl(A.ORI_HOLD_BALANCE, 0), '3',nvl(A.ORI_HOLD_BALANCE, 0) - nvl(A.ORI_REINVEST_BALANCE, 0)) > B.MIN_HOSTING AND decode(b.OPERATE_WAY, '0',a.balance, '1',nvl(A.balance, 0) - nvl(A.REINVEST_BALANCE, 0), '2',nvl(A.ORI_HOLD_BALANCE, 0), '3',nvl(A.ORI_HOLD_BALANCE, 0) - nvl(A.ORI_REINVEST_BALANCE, 0)) <= B.MAX_HOSTING and DECODE(B.AREA_FLAG,'2', a.HOLD_RATIO, '1', DECODE(b.OPERATE_WAY, '0', SHARES, '1', A.SHARES - A.REINVEST_SHARE, '2', SHARES, '3', A.SHARES - A.REINVEST_SHARE), DECODE(b.OPERATE_WAY, '0', A.BALANCE, '1', A.BALANCE - A.REINVEST_BALANCE, '2', a.ORI_HOLD_BALANCE, '3', A.ORI_HOLD_BALANCE - A.ORI_REINVEST_BALANCE)) >= B.MIN_HOSTING and DECODE(B.AREA_FLAG,'2', a.HOLD_RATIO, '1', DECODE(b.OPERATE_WAY, '0', SHARES, '1', A.SHARES - A.REINVEST_SHARE, '2', SHARES, '3', A.SHARES - A.REINVEST_SHARE), DECODE(b.OPERATE_WAY, '0', A.BALANCE, '1', A.BALANCE - A.REINVEST_BALANCE, '2', a.ORI_HOLD_BALANCE, '3', A.ORI_HOLD_BALANCE - A.ORI_REINVEST_BALANCE)) < B.MAX_HOSTING and not exists( select * from GRADING_FUND_PLAN tsc where tsc.MAIN_FUND_CODE = A.fund_code and tsc.SECTION_FUND_CODE = '******' ) and a.transaction_cfm_date >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.transaction_cfm_date <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} {"origin" :"param", "field":"agencyCodeList", "where" :" and a.distributor_code in (%s)"} {"origin" :"param", "field" :"fundCodeList", "where" :" and a.fund_code in (%s)"} /** and #WhereSql **/ ) hz , (select case when substr({"origin" :"param", "field" :"dateStart","where" :"%s"}, 1, 4)=substr({"origin" :"param", "field" :"dateEnd","where" :"%s"}, 1, 4) and mod(substr({"origin" :"param", "field" :"dateEnd","where" :"%s"}, 1, 4),4)=0 then '366' else '365' end YearDays from dual) year where GetSysValueRX('System','TailSegment','0')=0 and GetSysValueRX('System','TailMode','1')=1 group by distributor_code,fund_code,share_class,TRANSACTION_CFM_DATE union all select distributor_code, fund_code, TRANSACTION_CFM_DATE, share_class, sum(BALANCE) BALANCE, sum(SHARES) SHARES, sum(round(BALANCE / (case when YEAR_DAY>0 then YEAR_DAY else year.YearDays end) * nvl(TAIL_COMMISSION_RATIO, 0), 2)) fee, max(nvl(TAIL_COMMISSION_RATIO, 0)) TAIL_COMMISSION_RATIO, sum(nvl(income, 0)) income, avg(BALANCE) avgbalance, avg(SHARES) avgshares from( SELECT a.distributor_code, a.fund_code, 'A' share_class, a.TRANSACTION_CFM_DATE, decode(e.OPERATE_WAY, '0',a.balance, '1',nvl(a.balance, 0) - nvl(a.REINVEST_BALANCE, 0), '2',nvl(a.ORI_HOLD_BALANCE, 0), '3',nvl(a.ORI_HOLD_BALANCE, 0) - nvl(a.ORI_REINVEST_BALANCE, 0)) BALANCE, decode(e.OPERATE_WAY, '0',a.shares, '1',nvl(a.shares, 0) - nvl(a.REINVEST_SHARE, 0), '2',nvl(a.shares, 0), '3',nvl(a.shares, 0) - nvl(a.REINVEST_SHARE, 0)) SHARES, E.TAIL_COMMISSION_RATIO, a.income, nvl(e.YEAR_DAY,0) YEAR_DAY FROM( select a.distributor_code, a.fund_code, 'A' share_class, a.TRANSACTION_CFM_DATE, a.HOLD_DATE, a.HOLD_RATIO, nvl(a.income, 0) income, nvl(a.REINVEST_SHARE, 0) REINVEST_SHARE, nvl(a.REINVEST_BALANCE, 0) REINVEST_BALANCE, nvl(a.ORI_HOLD_BALANCE, 0) ORI_HOLD_BALANCE, NVL(A.ORI_REINVEST_BALANCE, 0) ORI_REINVEST_BALANCE, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_BALANCE, 0) else nvl(a.HOLD_BALANCE, 0) + nvl(a.income, 0) end balance, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_SHARE, 0) else nvl(a.HOLD_SHARE, 0) + nvl(a.income, 0) end shares from FUND_SALE_STAT a,fund_info b WHERE a.INDIVIDUAL_OR_INSTITUTION = '*' and A.TRANSACTION_CFM_DATE >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.TRANSACTION_CFM_DATE <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} and a.fund_code = b.fund_code ) a, vtailratio_tmp tfa, (select A.TRANSACTION_CFM_DATE,b.TAIL_COMMISSION_RATIO, a.distributor_code, a.fund_code, B.ENABLE_DATE, B.end_date, b.YEAR_DAY, b.OPERATE_WAY from( select a.distributor_code, a.fund_code, 'A' share_class, a.TRANSACTION_CFM_DATE, a.HOLD_DATE, a.HOLD_RATIO, nvl(a.income, 0) income, nvl(a.REINVEST_SHARE, 0) REINVEST_SHARE, nvl(a.REINVEST_BALANCE, 0) REINVEST_BALANCE, nvl(a.ORI_HOLD_BALANCE, 0) ORI_HOLD_BALANCE, NVL(A.ORI_REINVEST_BALANCE, 0) ORI_REINVEST_BALANCE, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_BALANCE, 0) else nvl(a.HOLD_BALANCE, 0) + nvl(a.income, 0) end balance, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_SHARE, 0) else nvl(a.HOLD_SHARE, 0) + nvl(a.income, 0) end shares from FUND_SALE_STAT a,fund_info b WHERE a.INDIVIDUAL_OR_INSTITUTION = '*' and A.TRANSACTION_CFM_DATE >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.TRANSACTION_CFM_DATE <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} and a.fund_code = b.fund_code ) a, (select fund_code, distributor_code, AREA_FLAG, MIN_HOSTING, MAX_HOSTING, TAIL_COMMISSION_RATIO, ENABLE_DATE,OPERATE_WAY,YEAR_DAY, last_value(end_date)over(partition by fund_code, distributor_code, ENABLE_DATE order by end_date rows between unbounded preceding and unbounded following) end_date from( select fund_code, distributor_code, AREA_FLAG, MIN_HOSTING, MAX_HOSTING, TAIL_COMMISSION_RATIO, ENABLE_DATE,OPERATE_WAY,YEAR_DAY, nvl(lead(ENABLE_DATE)over(partition by fund_code, distributor_code order by ENABLE_DATE, MIN_HOSTING), '20991231') end_date from vtailratio_tmp where AREA_FLAG in ('0','1') ) ) b WHERE b.fund_code = a.fund_code and b.distributor_code = a.distributor_code and a.HOLD_DATE >= B.ENABLE_DATE and a.HOLD_DATE < B.end_date group by a.distributor_code, a.fund_code, B.AREA_FLAG, B.TAIL_COMMISSION_RATIO, B.MIN_HOSTING, B.MAX_HOSTING, B.ENABLE_DATE, B.end_date,A.TRANSACTION_CFM_DATE,B.YEAR_DAY,b.OPERATE_WAY having DECODE(B.AREA_FLAG,'2', avg(a.HOLD_RATIO), '1', AVG(DECODE(b.OPERATE_WAY, '0', SHARES, '1', A.SHARES - A.REINVEST_SHARE, '2', SHARES, '3', A.SHARES - A.REINVEST_SHARE)), AVG(DECODE(b.OPERATE_WAY, '0', A.BALANCE, '1', A.BALANCE - A.REINVEST_BALANCE, '2', a.ORI_HOLD_BALANCE, '3', A.ORI_HOLD_BALANCE - A.ORI_REINVEST_BALANCE))) >= B.MIN_HOSTING and DECODE(B.AREA_FLAG,'2', avg(a.HOLD_RATIO), '1', AVG(DECODE(b.OPERATE_WAY, '0', SHARES, '1', A.SHARES - A.REINVEST_SHARE, '2', SHARES, '3', A.SHARES - A.REINVEST_SHARE)), AVG(DECODE(b.OPERATE_WAY, '0', A.BALANCE, '1', A.BALANCE - A.REINVEST_BALANCE, '2', a.ORI_HOLD_BALANCE, '3', A.ORI_HOLD_BALANCE - A.ORI_REINVEST_BALANCE))) < B.MAX_HOSTING ) e WHERE a.distributor_code = e.distributor_code(+) and a.fund_code = e.fund_code(+) AND A.TRANSACTION_CFM_DATE = E.TRANSACTION_CFM_DATE and a.fund_code = tfa.fund_code and a.distributor_code = tfa.distributor_code and a.HOLD_DATE >= e.ENABLE_DATE(+) and a.HOLD_DATE < e.end_date(+) and not exists(select * from GRADING_FUND_PLAN tsc where tsc.MAIN_FUND_CODE = a.fund_code and tsc.SECTION_FUND_CODE = '******') and a.transaction_cfm_date >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.transaction_cfm_date <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} {"origin" :"param", "field":"agencyCodeList", "where" :" and a.distributor_code in (%s)"} {"origin" :"param", "field" :"fundCodeList", "where" :" and a.fund_code in (%s)"} /** and #WhereSql **/ ) hz, (select case when substr({"origin" :"param", "field" :"dateStart","where" :"%s"}, 1, 4)=substr({"origin" :"param", "field" :"dateEnd","where" :"%s"}, 1, 4) and mod(substr({"origin" :"param", "field" :"dateEnd","where" :"%s"}, 1, 4),4)=0 then '366' else '365' end YearDays from dual) year where GetSysValueRX('System','TailSegment','0')=0 and GetSysValueRX('System','TailMode','1')=2 group by distributor_code,fund_code,share_class,TRANSACTION_CFM_DATE union all select distributor_code, fund_code, TRANSACTION_CFM_DATE, share_class, BALANCE, SHARES, fee, TAIL_COMMISSION_RATIO, income, avgbalance, avgshares from( select distributor_code, fund_code, TRANSACTION_CFM_DATE, share_class, sum(BALANCE) BALANCE, sum(SHARES) SHARES, sum(nvl(( select round(sum(DECODE(AREA_FLAG,'2',ROUND(SUM(GREATEST(LEAST(m.MAX_HOSTING-m.MIN_HOSTING,HZA.BALANCE-m.MIN_HOSTING),0)*TAIL_COMMISSION_RATIO/(case when hzA.YEAR_DAY>0 then hzA.YEAR_DAY else year.YearDays end)),2), '1',ROUND(SUM(GREATEST(LEAST(m.MAX_HOSTING-m.MIN_HOSTING,HZA.SHARES-m.MIN_HOSTING),0)*hzA.NAV*TAIL_COMMISSION_RATIO/(case when hzA.YEAR_DAY>0 then hzA.YEAR_DAY else year.YearDays end)),2), ROUND(SUM(greatest(least(m.MAX_HOSTING - m.MIN_HOSTING,hzA.BALANCE- m.MIN_HOSTING),0) * TAIL_COMMISSION_RATIO / 365)),2)),2) from( select fund_code, distributor_code, AREA_FLAG, MIN_HOSTING, MAX_HOSTING, TAIL_COMMISSION_RATIO, ENABLE_DATE, last_value(end_date)over(partition by fund_code, distributor_code, ENABLE_DATE order by end_date rows between unbounded preceding and unbounded following) end_date from( select fund_code, distributor_code, AREA_FLAG, MIN_HOSTING, MAX_HOSTING, TAIL_COMMISSION_RATIO, ENABLE_DATE, nvl(lead(ENABLE_DATE)over(partition by fund_code, distributor_code order by ENABLE_DATE, MIN_HOSTING), '20991231') end_date from vtailratio_tmp where AREA_FLAG in ('0','1')) ) m, (select case when substr({"origin" :"param", "field" :"dateStart","where" :"%s"}, 1, 4)=substr({"origin" :"param", "field" :"dateEnd","where" :"%s"}, 1, 4) and mod(substr({"origin" :"param", "field" :"dateEnd","where" :"%s"}, 1, 4),4)=0 then '366' else '365' end YearDays from dual) year where distributor_code = hzA.distributor_code and fund_code = hzA.fund_code and hzA.HOLD_DATE >= ENABLE_DATE and hzA.HOLD_DATE < end_date GROUP BY AREA_FLAG),0)) fee, 0 TAIL_COMMISSION_RATIO, sum(nvl(income,0)) income, avg(BALANCE) avgbalance, avg(SHARES) avgshares from( SELECT A.HOLD_DATE, A.distributor_code, A.fund_code, 'A' share_class, A.TRANSACTION_CFM_DATE, decode(tfa.OPERATE_WAY, '0',a.balance, '1',nvl(A.balance, 0) - nvl(A.REINVEST_BALANCE, 0), '2',nvl(A.ORI_HOLD_BALANCE, 0), '3',nvl(A.ORI_HOLD_BALANCE, 0) - nvl(A.ORI_REINVEST_BALANCE, 0)) BALANCE, decode(tfa.OPERATE_WAY, '0',a.shares, '1',nvl(A.shares, 0) - nvl(A.REINVEST_SHARE, 0), '2',nvl(A.shares, 0), '3',nvl(A.shares, 0) - nvl(A.REINVEST_SHARE, 0)) SHARES, A.income ,D.NAV NAV, nvl(tfa.YEAR_DAY,0) YEAR_DAY FROM( select a.distributor_code, a.fund_code, 'A' share_class, a.TRANSACTION_CFM_DATE, a.HOLD_DATE, a.HOLD_RATIO, nvl(a.income, 0) income, nvl(a.REINVEST_SHARE, 0) REINVEST_SHARE, nvl(a.REINVEST_BALANCE, 0) REINVEST_BALANCE, nvl(a.ORI_HOLD_BALANCE, 0) ORI_HOLD_BALANCE, NVL(A.ORI_REINVEST_BALANCE, 0) ORI_REINVEST_BALANCE, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_BALANCE, 0) else nvl(a.HOLD_BALANCE, 0) + nvl(a.income, 0) end balance, case when b.ACC_INCOME_TO_SHARE_FLAG = '0' or /** #TACode || **/ '94'||a.distributor_code = '48002' then nvl(a.HOLD_SHARE, 0) else nvl(a.HOLD_SHARE, 0) + nvl(a.income, 0) end shares from FUND_SALE_STAT a,fund_info b WHERE a.INDIVIDUAL_OR_INSTITUTION = '*' and A.TRANSACTION_CFM_DATE >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.TRANSACTION_CFM_DATE <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} and a.fund_code = b.fund_code ) A, vtailratio_tmp tfa , NET_VALUE D where A.fund_code = tfa.fund_code AND A.distributor_code = tfa.distributor_code AND A.fund_code = D.fund_code AND D.TRANSACTION_CFM_DATE = (select min(fd.TRANSACTION_CFM_DATE) from NET_VALUE fd where fd.TRANSACTION_CFM_DATE >= a.TRANSACTION_CFM_DATE and fd.fund_code = a.fund_code ) and not exists(select * from GRADING_FUND_PLAN tsc where tsc.MAIN_FUND_CODE = A.fund_code and tsc.SECTION_FUND_CODE = '******') and a.transaction_cfm_date >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.transaction_cfm_date <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} {"origin" :"param", "field":"agencyCodeList", "where" :" and a.distributor_code in (%s)"} {"origin" :"param", "field" :"fundCodeList", "where" :" and a.fund_code in (%s)"} /** and #WhereSql **/ ) hzA group by distributor_code,fund_code,share_class,TRANSACTION_CFM_DATE ) where GetSysValueRX('System', 'TailSegment', '0') = '1' AND GetSysValueRX('System', 'TailMode', '1') = '1' union all select A.distributor_code, A.fund_code, A.TRANSACTION_CFM_DATE,'A' share_class, 0 BALANCE, 0 SHARES, round(nvl(A.TRANSFER_FEE,0)*tfb.distributor_ratio,2) fee, 0 TAIL_COMMISSION_RATIO, 0 income, 0 avgbalance, 0 avgshares from( select t.* from( select * from trade_confirm union all select * from TRADE_CONFIRM_LOG ) t ) A,fee_belong tfb where A.fund_code = tfb.fund_code and A.distributor_code = tfb.distributor_code and tfb.fee_type = '31' and A.CHECK_RESULT = '1' and A.BUSINESS_CODE = '124' and A.TRANSFER_FEE>0 and GetSysValueRX('Report','TailsIncludeProfit','0')='1' /** and #WhereSql **/ and a.transaction_cfm_date >= {"origin" :"param", "field" :"dateStart","where" :"%s"} and a.transaction_cfm_date <= {"origin" :"param", "field" :"dateEnd","where" :"%s"} {"origin" :"param", "field":"agencyCodeList", "where" :" and a.distributor_code in (%s)"} {"origin" :"param", "field" :"fundCodeList", "where" :" and a.fund_code in (%s)"} ) group by distributor_code,fund_code,TRANSACTION_CFM_DATE )a, (select fund_code,distributor_code, min_tail_fare from( select fund_code,distributor_code,nvl(min_tail_fare,0) min_tail_fare, ROW_NUMBER() OVER(PARTITION BY fund_code , distributor_code ORDER BY fund_code DESC,distributor_code DESC) RANK from vtailratio_tmp) tfa where RANK=1 ) tfa,fund_info fi,distributor_info di where a.fund_code = tfa.fund_code and a.fund_code = fi.fund_code and a.distributor_code = di.distributor_code and a.distributor_code = tfa.distributor_code group by a.distributor_code,di.distributor_name,a.fund_code,fi.fund_name,tfa.min_tail_fare ,TAIL_COMMISSION_RATIO )order by fundcode,distributor_code给这段sql划分层次使人更容易理解
最新发布
07-16
<think>我们面对一个复杂SQL查询的优化问题,目标是提高可读性和可维护性。根据引用[2],我们需要关注SQL脚本的可读性和可维护性,包括合理的缩进、注释和命名规范。此外,引用[1]提到子查询可以用于构建层次化查询,引用[3]则强调了代码复用的重要性。 解决方案: 1. 使用CTE(公用表表达式)代替嵌套子查询:将复杂的子查询分解为命名的临时结果集,使主查询更清晰。 2. 合理使用缩进和注释:对每个部分进行注释,并保持一致的缩进风格。 3. 将重复逻辑抽取为视图或CTE:如果多个查询有相同部分,可创建视图,避免重复(引用[3]的复用思想)。 4. 避免过长的行:将长条件或表达式分行显示,并适当缩进。 5. 使用表别名简化引用,但别名要有意义。 示例:假设原查询是一个多层嵌套的子查询,我们可以将其改写为CTE形式。 原始查询可能类似: ```sql SELECT a, b, c FROM ( SELECT a, b, c FROM table1 WHERE condition1 ) AS sub1 JOIN ( SELECT d, e, f FROM table2 WHERE condition2 ) AS sub2 ON sub1.a = sub2.d WHERE sub1.b > 100; ``` 优化后使用CTE: ```sql WITH cte1 AS ( SELECT a, b, c FROM table1 WHERE condition1 ), cte2 AS ( SELECT d, e, f FROM table2 WHERE condition2 ) SELECT cte1.a, cte1.b, cte1.c, cte2.e, cte2.f FROM cte1 JOIN cte2 ON cte1.a = cte2.d WHERE cte1.b > 100; ``` 这样,每个子查询被拆分成独立的CTE,主查询变得简洁明了。 另外,对于非常复杂的查询,我们可以考虑将部分逻辑封装成视图,这样不仅可以在当前查询中使用,还可以被其他查询复用(引用[3]的代码复用思想)。 步骤总结: 1. 识别嵌套子查询或重复部分。 2. 将每个子查询转换为CTE。 3. 使用有意义的CTE名称。 4. 添加注释说明每个CTE的作用。 5. 保持一致的缩进(通常每个关键字如SELECT、FROM等换行并缩进)。 注意:CTE在大多数数据库(如PostgreSQL, SQL Server, Oracle, BigQuery等)中都支持,但MySQL 8.0之前版本不支持CTE,此时可以考虑创建临时表或视图。 如果查询中包含重复使用的子查询部分,可以创建视图,例如: ```sql CREATE VIEW view1 AS SELECT a, b, c FROM table1 WHERE condition1; ``` 然后在查询中直接使用视图。 最后,引用[2]提到,合理的命名规范也很重要,确保CTE、视图、列等命名具有描述性。 因此,对于用户的问题,我们可以这样回答:</think>### 如何优化复杂SQL查询的层次结构 根据引用[2]和[3]的核心观点,优化复杂SQL查询的关键在于**增强可读性**和**提升可维护性**。以下是分步优化方案: #### 1. **使用CTE分解嵌套查询**(解决多层嵌套问题) ```sql WITH department_stats AS ( -- 注释:计算部门级统计 SELECT dept_id, AVG(salary) AS avg_salary, COUNT(*) AS emp_count FROM employees WHERE hire_date > '2020-01-01' GROUP BY dept_id ), project_budgets AS ( -- 注释:获取项目预算数据 SELECT project_id, SUM(budget) AS total_budget FROM projects WHERE status = 'ACTIVE' GROUP BY project_id ) -- 主查询(清晰关联逻辑) SELECT d.dept_name, ds.avg_salary, pb.total_budget FROM departments d JOIN department_stats ds ON d.id = ds.dept_id JOIN dept_projects dp ON d.id = dp.dept_id JOIN project_budgets pb ON dp.project_id = pb.project_id WHERE ds.emp_count > 5; ``` **优势**: - 将嵌套子查询转为命名模块(CTE) - 每个CTE有明确功能注释 - 主查询仅保留关联逻辑[^1] #### 2. **视图封装复用逻辑**(解决代码重复问题) ```sql CREATE VIEW v_employee_performance AS -- 复用视图 SELECT e.id, e.name, SUM(s.amount) AS total_sales, RANK() OVER(ORDER BY SUM(s.amount) DESC) AS rank FROM employees e JOIN sales s ON e.id = s.emp_id GROUP BY e.id, e.name; ``` **应用场景**: ```sql -- 多场景复用视图 SELECT * FROM v_employee_performance WHERE rank <= 10; SELECT dept_id, AVG(total_sales) FROM v_employee_performance GROUP BY dept_id; ``` **价值**:避免引用[3]中描述的"相同逻辑散落各处"问题[^3] #### 3. **分层缩进与注释规范** ```sql /* 目标:获取高绩效部门的项目预算 层次: 1. 部门绩效数据 → 2. 项目预算 → 3. 最终关联 */ SELECT ... -- 主查询 FROM ( -- 层级1:部门绩效子查询 SELECT ... FROM ( -- 层级2:员工绩效基础数据 SELECT ... FROM employees ) emp_perf ) dept_perf JOIN ... -- 关联逻辑 ``` #### 4. **关键优化策略对比** | 方法 | 适用场景 | 优势 | 注意事项 | |---------------|-------------------------|-----------------------------|---------------------| | **CTE** | 单次查询的模块化 | 逻辑隔离、按顺序执行 | 避免过度拆分(>5个CTE) | | **视图** | 跨查询复用逻辑 | 统一维护、减少重复代码 | 需权限管理 | | **临时表** | 大数据集分阶段处理 | 提升性能、支持索引 | 增加I/O开销 | ### 实施建议 1. **渐进式重构**:先对最复杂的子查询进行CTE拆分 2. **命名规范**:使用`业务域_功能`格式(如`sales_daily_summary`) 3. **工具辅助**:利用SQL Formatter工具统一缩进(如pgFormatter) 4. **文档化**:对每个CTE/视图添加元数据注释: ```sql COMMENT ON VIEW v_employee_performance IS '计算员工销售绩效排名,包含:员工ID、姓名、销售总额、排名'; ``` > **经验提示**:当单个查询超过200行或包含3层以上嵌套时(如引用[3]描述的机关级次计算场景),必须进行结构化拆分[^3]。每次修改后,使用`EXPLAIN`验证执行计划是否优化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值