原来rank() over()函数不止排序

简介: 原来rank() over()函数不止排序

最近在做一个需求,其中有部分逻辑,秉着一切皆可left join 解决的方式写完了,直到我师兄说我这部分代码可以换个方式写的时候,我:哇,原来还可以这样处理。

已知数据集:

1ecd1b2606ed46e9956a89f231c9802c.png

取数要求:当uid的tag为人称、IP 时,则保留tag为人称、IP时的全部记录(意味着,剔除tag为领域的记录);当uid的tag没有人称、IP时,则取tag为领域的全部记录。

首先看我的代码:

SELECT  T.uid
       ,T.tag
       ,T.tag_name
FROM
(
    SELECT  *
    FROM T
    WHERE tag = '领域' 
) T
LEFT JOIN
(
    SELECT  uid
           ,tag
           ,tag_name
    FROM T
    WHERE tag IN ('人称', 'IP') 
) T1
ON T.uid = T1.uid
WHERE T1.uid is null 
UNION ALL
SELECT  uid
       ,tag
       ,tag_name
FROM T
WHERE tag IN ('人称', 'IP') ;

结果:

1ecd1b2606ed46e9956a89f231c9802c.png

 我的逻辑就是先把所有tag='领域'和tag为人称和IP的记录分别挑出来,用uid去left join 就可以剔除拥有tag人称、IP、领域,tag='领域'的记录,只保留单纯拥有领域的记录,再和拥有人称领域的部分union。(只能说结果虽对,但看起来傻傻的)

使用rank() over()函数

       这个函数本身是用来分组排序的,在这里先给tag(文本)一个数字标识,要取的内容打成相同数字标识,不取的打另外的数字标识,再通过分组倒序排序(按数字标识)就可以很容易的剔除不想要的记录了。

SELECT  t3.uid
       ,t3.tag
       ,t3.tag_name
FROM
(
  SELECT  t2.uid
         ,t2.tag
         ,t2.tag_name
         ,rank() over(partition by uid ORDER BY rn desc) AS rank
  FROM
  (
    SELECT  T.uid
           ,T.tag
           ,T.tag_name
           ,CASE WHEN tag = '领域' THEN 1  ELSE 2 END AS rn
    FROM T
  )t2
) t3
WHERE t3.rank = 1

结果:

1ecd1b2606ed46e9956a89f231c9802c.png

       rank() over()函数在这里起到了辅助剔除记录的作用。  

相关文章
|
8月前
lambda中sorted排序
lambda中sorted排序
|
搜索推荐 C++
C++利用sort进行排序
C++利用sort进行排序
|
8月前
|
搜索推荐 数据库 C++
带用排序等法sort讲解
带用排序等法sort讲解
51 0
|
8月前
|
C++
C++如何进行sort的使用——C++如何进行排序
C++如何进行sort的使用——C++如何进行排序
157 0
|
8月前
|
小程序
排序sort()排序用法
排序sort()排序用法
|
8月前
ROW_NUMBER() OVER()函数用法详解 (分组排序 例子多)
ROW_NUMBER() OVER()函数用法详解 (分组排序 例子多)
127 0
排序(Sort)(一)
排序(Sort)(一)
91 0
排序(Sort)(二)
排序(Sort)(二)
72 0
rank()、dense-rank()、row-number()的区别
rank()、dense-rank()、row-number()的区别
126 0
|
SQL Oracle 关系型数据库
Oracle-分析函数之排序值rank()和dense_rank()
Oracle-分析函数之排序值rank()和dense_rank()
175 0