分享好友 最新动态首页 最新动态分类 切换频道
转载oracle排序排名写法
2024-12-27 09:59

(1)rank函数返回一个唯一的值,除非遇到相同的数据时,此时所有相同数据的排名是一样的,同时会在最后一条相同记录和下一条不同记录的排名之间空出排名。 
(2)dense_rank函数返回一个唯一的值,除非当碰到相同数据时,此时所有相同数据的排名都是一样的。
(3)row_number函数返回一个唯一的值,当碰到相同数据时,排名按照记录集中记录的顺序依次递增。
(4)ntile是要把查询得到的结果平均分为几组,如果不平均则分给第一组。

转载oracle排序排名写法

例如

create table s_score
(   s_id number(6)
   ,score number(4,2)
);
insert into s_score values(001,98);
insert into s_score values(002,66.5);
insert into s_score values(003,99);
insert into s_score values(004,98);
insert into s_score values(005,98);
insert into s_score values(006,80);

select
    s_id 
   ,score
   ,rank() over(order by score desc) rank               --按照成绩排名,纯排名
   ,dense_rank() over(order by score desc) dense_rank   --按照成绩排名,相同成绩排名一致
   ,row_number() over(order by score desc) row_number   --按照成绩依次排名
   ,ntile(3) over (order by score desc) group_s         --按照分数划分成绩梯队
from s_score;

排名/排序的时候,有时候,我们会想到利用伪列row_num,利用row_num确实可以解决某些场景下的问题(但是相对也比较复杂),而且有些场景下的问题却很难解决。

例:取成绩前三名,并且前三名含有并列的情况。通过上面例子,我们可以直观的看到,结果应该有5条记录

select
    s_id 
   ,score
   ,dense_rank
from (
select
    s_id 
   ,score
   ,rank() over(order by score desc) rank
   ,dense_rank() over(order by score desc) dense_rank
   ,row_number() over(order by score desc) row_number
from s_score
) t
where dense_rank <= 3;

   S_ID  SCORE DENSE_RANK
------- ------ ----------
      3  99.00          1
      1  98.00          2
      5  98.00          2
      4  98.00          2
      6  80.00          3
      

如果只是简单的想到去用rownum <= 3 得到的结果显然不可能是正确的。

组内的排名或者排序是经常遇到的一种场景。
例如,取每个销售部门内,销售业绩最好的前三名。取每个班级内成绩排名信息等等..
取每个班级内每门课成绩排名第一的同学信息

drop table S_SCORE;
create table S_SCORE
(
  S_ID  NUMBER(6),
  CLASS_ID VARCHAR2(2),
  COURSE VARCHAR2(20),
  SCORE NUMBER(5,2)
);

INSERT INTO S_SCORE VALUES(1001,'A','MATH','67');
INSERT INTO S_SCORE VALUES(1004,'B','MATH','88');
INSERT INTO S_SCORE VALUES(1002,'A','MATH','99');
INSERT INTO S_SCORE VALUES(1003,'A','MATH','55');
INSERT INTO S_SCORE VALUES(1001,'B','MATH','88');
INSERT INTO S_SCORE VALUES(1001,'B','MATH','70');
INSERT INTO S_SCORE VALUES(1001,'A','ORACLE','97');
INSERT INTO S_SCORE VALUES(1004,'B','ORACLE','48');
INSERT INTO S_SCORE VALUES(1002,'A','ORACLE','79');
INSERT INTO S_SCORE VALUES(1003,'A','ORACLE','65');
INSERT INTO S_SCORE VALUES(1001,'B','ORACLE','82');
INSERT INTO S_SCORE VALUES(1001,'B','ORACLE','78');

select
   s_id
  ,class_id
  ,course
  ,score
  ,dense_rank() over (partition by class_id,course order by score desc) drk
from S_SCORE;

   S_ID CLASS_ID COURSE                 SCORE        DRK
------- -------- -------------------- ------- ----------
   1002 A        MATH                   99.00          1
   1001 A        MATH                   67.00          2
   1003 A        MATH                   55.00          3
   1001 A        ORACLE                 97.00          1
   1002 A        ORACLE                 79.00          2
   1003 A        ORACLE                 65.00          3
   1004 B        MATH                   88.00          1
   1001 B        MATH                   88.00          1
   1001 B        MATH                   70.00          2
   1001 B        ORACLE                 82.00          1
   1001 B        ORACLE                 78.00          2
   1004 B        ORACLE                 48.00          3
   
select
   s_id
  ,class_id
  ,course
  ,score
from (
select
   s_id
  ,class_id
  ,course
  ,score
  ,dense_rank() over (partition by class_id,course order by score desc) drk
from S_SCORE
) t
where drk = 1;


   S_ID CLASS_ID COURSE                 SCORE
------- -------- -------------------- -------
   1002 A        MATH                   99.00
   1001 A        ORACLE                 97.00
   1004 B        MATH                   88.00
   1001 B        MATH                   88.00
   1001 B        ORACLE                 82.00   

rank()和dense_rank()用法相似,这里就不在举例说明了。可以将上面的例子中dense_rank()替换成rank()实现。

接下来,看一个使用row_number()的场景
例:查看每个部门最近一笔销售记录

select * from criss_sales order by dept_id,sale_date desc;

DEPT_ID SALE_DATE   GOODS_TYPE    SALE_CNT
------- ----------- ---------- -----------
D01     2014/5/4    G02                 80
D01     2014/4/30   G03                800
D01     2014/4/8    G01                200
D01     2014/3/4    G00                700
D02     2014/5/2    G03                900
D02     2014/4/27   G01                300
D02     2014/4/8    G02                100
D02     2014/3/6    G00                500

即,我们希望得到这两条记录

D01     2014/5/4    G02                 80
D02     2014/5/2    G03                900

select
  dept_id
 ,sale_date
 ,goods_type
 ,sale_cnt
 ,row_number() over (partition by dept_id order by sale_date desc)
from criss_sales;

DEPT_ID SALE_DATE   GOODS_TYPE    SALE_CNT ROW_NUMBER()OVER(PARTITIONBYDE
------- ----------- ---------- ----------- ------------------------------
D01     2014/5/4    G02                 80                              1
D01     2014/4/30   G03                800                              2
D01     2014/4/8    G01                200                              3
D01     2014/3/4    G00                700                              4
D02     2014/5/2    G03                900                              1
D02     2014/4/27   G01                300                              2
D02     2014/4/8    G02                100                              3
D02     2014/3/6    G00                500                              4

select
  dept_id
 ,sale_date
 ,goods_type
 ,sale_cnt
from (
select
  dept_id
 ,sale_date
 ,goods_type
 ,sale_cnt
 ,row_number() over (partition by dept_id order by sale_date desc) rn
from criss_sales
) t
where rn = 1;

DEPT_ID SALE_DATE   GOODS_TYPE    SALE_CNT
------- ----------- ---------- -----------
D01     2014/5/4    G02                 80
D02     2014/5/2    G03                900

 

有时会有这样的需求:如果数据排序后分为三部分,业务人员只关心其中的一部分,如何将这中间的三分之一数据拿出来呢?
这时比较好的选择,就是使用ntile函数

select
      dept_id
     ,sale_date
     ,goods_type
     ,sale_cnt
     ,ntile(3) over (order by sale_cnt desc nulls last)  all_cmp
     ,ntile(3) over (partition by dept_id order by sale_cnt desc nulls last) all_dept
from criss_sales;

最新文章
智能制造系统与先进设备高效生产线的核心技术
智能制造系统与先进设备(高效生产线的核心技术)什么是智能制造?智能制造是一种集成了现代信息技术和传统制造技术的生产方式,它通过在产品设计、原材料采购、生产过程控制、质量检测以及物流配送等环节采用自动化和信息化手段,提高了生
如何设计一个高逼格的LOGO ?强推AI免费LOGO生成神器!
在商业世界里,每一个成功品牌背后都有一段值得讲述的故事。比如苹果公司,从乔布斯和沃兹尼亚克在车库里起步,到如今成为全球最有价值的品牌之一,其标志性的苹果logo见证了这一切。这个简单的logo不仅仅是一个图形,它承载着企业的灵魂,
网站推广渠道有哪些,网站推广渠道有哪些
成都本站公司长期为近1000家客户提供的网站建设服务,团队从业经验10年,关注不同地域、不同群体,并针对不同对象提供差异化的产品和服务;打造开放共赢平台,与合作伙伴共同营造健康的互联网生态环境。为单县企业提供专业的网站设计、成都
百度贴吧排名规则与蜘蛛池,优化策略与影响因素解析,百度贴吧帖子排名
百度贴吧的排名规则主要基于内容质量、用户互动、账号权重等因素。优化策略包括发布高质量内容、增加用户互动、提高账号权重等。蜘蛛池是一种工具,可以帮助提高网站或帖子在搜索引擎中的排名。影响因素包括关键词密度、标题标签、内容原创
看过来,这里有小红书SEO与品牌推广全攻略
随着社交媒体领域的蓬勃发展和数字营销策略的不断精进,小红书(Red),这一中国顶 尖的生活方式分享与电商平台,正吸引着众多品牌及营销精英的深切关注。利用小红书进行搜索引擎优化(SEO)与品牌推广,是品牌在海量用户群体中崭露头角、
爱熊本熊品牌怎么样申请店铺
在奶茶市场都在追求创意和爆点的当下,唯有有特色、有卖点、有创意的品牌才能得到消费者的喜爱、创业者的认可!在创意潮流中,跨界似乎成为了今年奶茶加盟市场里最流行的方式,奶茶与动漫的跨界合作尤为明显!南京独道餐饮管理有限公司成功获
提升网站排名的技巧与策略:如何通过百度SEO优化实现流量增长
随着互联网的发展,越来越多的企业和个人开始关注搜索引擎优化(SEO)这一领域,尤其是百度SEO优化,因为百度是中国最大的搜索引擎,几乎占据了国内搜索市场的绝大部分份额。因此,如何通过百度SEO优化提升网站排名,吸引更多流量,成为了
seo从入门到精通_SEO优化从入门到精通的十大技巧
入门级的SEO学习方法,是学习SEO主要的流程和步骤。赢客互动seo认为只要以下10个技巧,就能让你SEO优化从入门到精通。2、了解服务器域名备案,如何注册域名,如何去解析域名,如何维护域名的安全,域
华为Mate 70/X6系列水冷壳上市,助力麒麟9020性能稳定持久!
华为最新旗舰手机Mate 70与X6系列在万众瞩目中正式上市,与此同时,一款创新的微泵液冷手机壳也同步亮相,为手机性能保驾护航。Mate 70系列手机壳定价亲民,为299元,而X6系列则稍高,售价为399元。这款手机壳的核心亮点在于其强大的功能性
相关文章
推荐文章
发表评论
0评