首頁 >資料庫 >mysql教程 >Oracle 中的 RANK() 與 DENSE_RANK():這些視窗函數在排序和處理空值上有何不同?

Oracle 中的 RANK() 與 DENSE_RANK():這些視窗函數在排序和處理空值上有何不同?

Mary-Kate Olsen
Mary-Kate Olsen原創
2025-01-25 00:12:10207瀏覽

RANK() vs. DENSE_RANK() in Oracle: How Do These Window Functions Differ in Ranking and Handling Nulls?

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 LASTORDER BY 規格。 此範例強調了在根據您的特定需求選擇適當的排名函數時了解這些細微差別的重要性。

以上是Oracle 中的 RANK() 與 DENSE_RANK():這些視窗函數在排序和處理空值上有何不同?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn