1 小時 SQL 極速入門(三)——分析函數

来源:https://www.cnblogs.com/injet/archive/2018/12/15/10122832.html
-Advertisement-
Play Games

1 小時 SQL 極速入門 前面兩篇我們從 SQL 的最基礎語法講起,到表聯結多表查詢。 大家可以點擊鏈接查看 "1 小時 SQL 極速入門(一)" "1 小時 SQL 極速入門(二)" 今天我們講一些在做報表和複雜計算時非常實用的分析函數。由於各個資料庫函數的實現不太一樣,本文基於 Oracle ...


1 小時 SQL 極速入門

前面兩篇我們從 SQL 的最基礎語法講起,到表聯結多表查詢。
大家可以點擊鏈接查看
1 小時 SQL 極速入門(一)
1 小時 SQL 極速入門(二)
今天我們講一些在做報表和複雜計算時非常實用的分析函數。由於各個資料庫函數的實現不太一樣,本文基於 Oracle 12c 。

ROW_NUMBER()函數

這個函數在平時用的還是比較多的。這個函數的作用是為分組內的每一行返回一個行號。我們還是舉例來說明。
假設我們有以下數據表:

ORDER.png

共 8 個訂單,分為 A,B,C,D四種類型,後面兩列是訂單描述和訂單數量。

假如我們現在想找到每個訂單類型中數量最少的一行記錄,比如想找到 A 類型訂單數量最少的,B 類型訂單數量最少的。。。
我們要怎麼寫呢 ? 用 GROUP BY 可能會很麻煩。這裡用 ROW_NUMBER() 就很合適

SELECT order_no,
  order_type,
  order_text,
  order_qty,
  row_number() OVER(PARTITION BY order_type order by order_qty) AS rowno
FROM wip_order_test

結果:
ROW_NUMBER.png

可以看到,每一行最後都有一個從低到高的編號,有了這個編號我們就可以通過取編號為 1 的行來得到每個分組中訂單數量最少的一行記錄。

解釋一下,ROW_NUMBER() 為每一行返回一個行號, partition by 表示分組,這裡表示根據 order_type 分組,然後我們按照訂單數量排序。就會得到每個分組內的按照訂單數量排序的行號。

SUM() OVER()函數

假如我們現在要 查詢每個類型的訂單總數分別是多少,要怎麼做?
大家可能會想到 GROUP BY,不過大家可以自己試試,是否能得到和我同樣的結果

SELECT order_no,
  order_type,
  order_text,
  order_qty,
  sum(order_qty) OVER(PARTITION BY order_type) AS sum_qty
FROM wip_order_test

結果:
SUM.png

看到後面多了一個數量列,就是每個分組的訂單總數量。是不是很方便?

除了 SUM 函數,其他幾個計算函數如 AVG(),MAX(),MIN(),COUNT()的使用方法和 SUM 一樣。

視窗函數

視窗函數可以對一個結果集內的一定範圍內值進行累積,或者通過移動視窗進行累積。還是看例子吧。

SELECT order_no,
  order_type,
  order_text,
  order_qty,
  sum(order_qty) OVER
    (ORDER BY order_no ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW)
    AS cumulative_qty
FROM wip_order_test;

視窗.png

解釋一下:還是用 SUM 來計算總和,這裡我們使用了新的語法, ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW 定義了視窗的起點和終點,UNBOUNDED PRECEDING表示起點在第一行,CURRENT ROW 表示終點在當前行。我們看一下上圖的結果,能看到最後一列的值是逐行累加的。

移動視窗

上面我們的視窗的起點是固定的,終點逐漸往下移,我們可以創建一個固定大小的視窗,起點和終點同時往下移動。只需要修改 UNBOUNDED 為一個固定的數字就可以了。我們修改成 2, 和 3 分別看一下

SELECT order_no,
  order_type,
  order_text,
  order_qty,
  SUM(order_qty) OVER (ORDER BY order_no ROWS BETWEEN 2 PRECEDING AND CURRENT ROW) AS cumulative_qty2,
  SUM(order_qty) OVER (ORDER BY order_no ROWS BETWEEN 3 PRECEDING AND CURRENT ROW) AS cumulative_qty3
FROM wip_order_test;

移動視窗.png

解釋下:倒數第二列我們修改視窗起點2,表示當前行與前兩行之間的範圍。可以看到每一行的值都是當前行與它前面兩行的值的累加。而最後一列,是當前行與它之前3行的值的累加。每處理一行,視窗的起點和終點都向下移動。

同理,SUM 也可以改為 AVG 求視窗的平均值

FIRST_VALUE() 和 LAST_VALUE()可以獲取視窗的第一行和最後一行,NTH_VALUE()可以獲取第 N 行。看一下例子:

SELECT order_no,
  order_type,
  order_text,
  order_qty,
  first_value(order_qty) OVER (ORDER BY order_no ROWS BETWEEN 2 PRECEDING AND CURRENT ROW) AS first_value,
  last_value(order_qty) OVER (ORDER BY order_no ROWS BETWEEN 3 PRECEDING AND CURRENT ROW)  AS last_value,
  nth_value(order_qty,2) OVER (ORDER BY order_no ROWS BETWEEN 3 PRECEDING AND CURRENT ROW) AS second_value
FROM wip_order_test;

FIRST_VALUE.png

LISTAGG() 函數

這個函數很有用,有時候在 GROUP BY 以後,我們想讓分組內的某一列的幾個值顯示在一行上,比如:

SELECT 
  order_type,
  listagg(to_char(order_text),'-') WITHIN GROUP (ORDER BY order_type) AS text
FROM wip_order_test
GROUP BY order_type

結果:
LISTAGG.png

看到,通過 LISTAGG ,把每個分組中的訂單描述欄位連接起來。第一個參數表示要合併的欄位名字,第二個參數表示分隔符。

TOP-N 查詢

Oracle 12c中新增了對 TOP-N的支持。

SELECT order_no,
  order_type,
  order_text,
  order_qty
FROM wip_order_test
FETCH FIRST 3 ROWS ONLY;

TOPN.png

我們用 FETCH FIRST 3 取出了前 3 行數據,這裡也可以使用 FETCH FIRST 20 PERCENT ROWS ONLY 用百分比來取出前 20% 的數據。

還可以使用 OFFSET 關鍵字,來表示從第幾行開始取,比如 OFFSET 5 ROWS FETCH NEXT 3 ROWS ONLY 就表示從第 5 行開始往下取 3 行。

中位數 PERCENTILE_CONT()

可以算一組值的中位數,傳入一個參數,比如傳入0.5 表示 1/2 中位數,0.75 表示 3/4 中位數

SELECT order_type,
  percentile_cont(0.5) WITHIN GROUP (
ORDER BY order_qty) AS A,
  percentile_cont(0.75) WITHIN GROUP (
ORDER BY order_qty) AS b
FROM wip_order_test
GROUP BY order_type

中位數.png

我們根據訂單類型分組後,分別算出每種訂單類型數量的 1/2 中位數和 3/4中位數。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 2.1 控制器的角色 MVC模式中的控制器(Controller)主要負責響應用戶的輸入,冰球在響應時修改模型(Model)。通過這種方式,MVC模式中的控制器主要關註的是應用程式流、輸入數據的處理,以及對相關視圖(View)輸出數據的提供。 URL首先告知陸游機制去實例化哪個控制器,調用哪個操作方 ...
  • C# winfrom 寫的一個搜索助手,可以按照標題和內容搜索,指定目錄後,遍歷搜索文件和子目,現在只寫了支持.DOC.DOCX.XLS.XLSX.PPT.PPTX.PDF.HTML.HTM.TXT等格式的文件搜索,讀取execl 內容使用的是NPOI組件,doc,html,txt 格式的使用Str ...
  • 我利用網路上的windows 10 純凈版來進行安裝windows 10 鏡像的時候,發現很多的windows 的服務都是不能用的。比如啟動/刪除 windows 功能就是不能用的,會出現如下信息: 發現出現了:“參照的程式集沒有安裝在系統上” 的錯誤,這應該是,你不小心安裝上精簡版系統造成的,某些 ...
  • 前幾節里,通過幾個案例,分析了各種常見的 CPU 性能問題。通過這些,相信對 CPU 的性能分析已經不再陌生和恐懼,起碼有了基本的思路,也瞭解了不少 CPU 性能的分析工 具。 不過,我猜你可能也碰到了一個我曾有過的困惑: CPU 的性能指標那麼多,CPU 性能分析工具 也是一抓一大把,如果離開專欄 ...
  • 控制台命令 運行redis # ./bin/redis-server ./redis.conf 搜索redis是否運行 # ps -ef | grep -i redis 關閉redis # ./bin/redis-cli shutdown 進入redis # ./bin/redis-cli 127. ...
  • 多表聯合查詢簡介 定義:連接查詢是關係型資料庫最主要的查詢,通過連接運算符可以實現多個表連接數據查詢。 分類:內連接,外連接,全外連接。 內連接 定義 內聯接使用比較運算符根據每個表的通用列中的值匹配兩個表中的行。 語法 select 表1.*,表2.* from 表1 as a [wtih(鎖)] ...
  • 問題描述:主庫備庫之前正常連接,但是昨天磁碟空間滿了之後,由於不知什麼原因將備庫重做日誌刪了,今天早上發現DG不同步的報警。 當時思路如下:1、通過select thread#,low_sequence#,high_sequence# from v$archive_gap; 查看是否有歸檔沒有傳到備 ...
  • 一. Hbase 的 region 我們先簡單介紹下 Hbase 的 架構和 region : 從物理集群的角度看,Hbase 集群中,由一個 Hmaster 管理多個 HRegionServer,其中每個 HRegionServer 都對應一臺物理機器,一臺 HRegionServer 伺服器上又 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...