Oracle RANK() 與 DENSE_RANK():視窗函數的比較分析
Oracle 的 RANK()
和 DENSE_RANK()
是視窗函數,用於為指定分區內的行分配排名。 雖然兩者都實現了相似的目標,但它們對關係和 NULL 值的處理卻截然不同。
RANK()
功能
RANK()
依照指定的順序分配排名。 至關重要的是,它將相同的排名分配給並列的行,然後跳過排名。 例如,如果三行並列第二名,則它們都將獲得 2 的排名,下一個排名將為 5。
DENSE_RANK()
功能
與RANK()
不同,DENSE_RANK()
分配連續的排名,沒有間隙,即使有平局。 所有並列的行都獲得相同的排名,隨後的排名緊隨其後。 沒有跳過的排名。
檢索第 n 個最高薪水
要確定 emptbl
表中的第 n 個最高工資,您可以在 RANK()
子句中使用 DENSE_RANK()
或 OVER()
:
<code class="language-sql">SELECT empname, sal FROM ( SELECT empname, sal, RANK() OVER (ORDER BY sal DESC) AS rnk -- Note: ORDER BY sal DESC for nth highest FROM emptbl ) WHERE rnk = n;</code>
處理 NULL 值
兩個函數對 NULL 值的處理由 ORDER BY
子句決定。 如果指定了 NULLS FIRST
,則 NULL 會排在非 NULL 值之前;如果NULLS LAST
,則排在後面。
說明性範例
讓我們檢查以下腳本:
<code class="language-sql">WITH q AS ( SELECT 10 AS DEPTNO, 'rrr' AS EMPNAME, 10000.00 AS SAL FROM DUAL UNION ALL SELECT 11, 'nnn', 20000.00 FROM DUAL UNION ALL SELECT 11, 'mmm', 5000.00 FROM DUAL UNION ALL SELECT 12, 'kkk', 30000 FROM DUAL UNION ALL SELECT 10, 'fff', 40000 FROM DUAL UNION ALL SELECT 10, 'ddd', 40000 FROM DUAL UNION ALL SELECT 10, 'bbb', 50000 FROM DUAL UNION ALL SELECT 10, 'xxx', NULL FROM DUAL UNION ALL SELECT 10, 'ccc', 50000 FROM DUAL ) SELECT EMPNAME, DEPTNO, SAL, RANK() OVER (PARTITION BY DEPTNO ORDER BY SAL NULLS FIRST) AS R, DENSE_RANK() OVER (PARTITION BY DEPTNO ORDER BY SAL NULLS FIRST) AS DR1, DENSE_RANK() OVER (PARTITION BY DEPTNO ORDER BY SAL NULLS LAST) AS DR2 FROM q;</code>
結果清楚地展示了 RANK()
和 DENSE_RANK()
在處理 NULL 值時的對比行為,取決於 NULLS FIRST
子句中的 NULLS LAST
或 ORDER BY
規格。 此範例強調了在根據您的特定需求選擇適當的排名函數時了解這些細微差別的重要性。
以上是Oracle 中的 RANK() 與 DENSE_RANK():這些視窗函數在排序和處理空值上有何不同?的詳細內容。更多資訊請關注PHP中文網其他相關文章!