sql根據團隊樹一級一級彙總統計

来源:https://www.cnblogs.com/wlc-atlantide/archive/2022/12/31/17016458.html
-Advertisement-
Play Games

​ 1、需求描述 最近碰到了一個需求,是要統計各個團隊的員工的銷售金額,然後一級一級向上彙總。 ​編輯 架構團隊樹是類似於這種樣子的,需要先算出每個員工的銷售金額,然後彙總成上一級的團隊金額,然後各個團隊的銷售總金額再往上彙總成一個區域的銷售金額,然後各個區域的金額再往上彙總成總公司的金額。當然我工 ...


 1、需求描述

        最近碰到了一個需求,是要統計各個團隊的員工的銷售金額,然後一級一級向上彙總。

編輯

 架構團隊樹是類似於這種樣子的,需要先算出每個員工的銷售金額,然後彙總成上一級的團隊金額,然後各個團隊的銷售總金額再往上彙總成一個區域的銷售金額,然後各個區域的金額再往上彙總成總公司的金額。當然我工作碰到的團隊樹要遠比這個複雜許多,但反正差不多是這麼個意思。

 

 

2、解決方法

2.1、方法一(不推薦)

        持久層通過一些sql把團隊樹結構,以及各個員工的銷售金額彙總拿到,然後在業務層通過代碼去一層層拼起來。這是我一開始拿到這個需求時的思路,後來發現可以但是很複雜,代碼可讀性及可維護性很差。

 

2.2、方法二(推薦)

        在sql裡面計算彙總出來。

        我這裡是在測試環境建了幾張Demo表來加以說明sql的邏輯。

1、建表、

CREATE TABLE Business..TGroupV2(TreeNodeNo int,TeamId int,TeamName varchar(100),[Path] varchar(100));
CREATE TABLE Business..TPeopleSalesInfoV2(TeamId int,PeopleId varchar(100),PeopleName varchar(100));
CREATE TABLE Business..TPeopleSalesDetailInfoV2(PeopleId varchar(100),Amount Decimal(18,2),ContractID varchar(100));

2、添加一些測試數據

編輯

 

3、SQL代碼

--以團隊為單位,彙總各個團隊,子團隊,父團隊的銷售金額
SELECT TB.TreeNodeNo,TB.TeamID,TB.TeamName,AA.Amount,'' as PeopleId ,'' as PeopleName FROM 
(
	SELECT A.ParentTeamID,SUM(A.Amount) as Amount FROM
		(
		SELECT  
			TT.*,TG2.TeamID as ParentTeamID,BB.Amount from
				(
					select T1.*,TG.[Path]
					from Business..TPeopleSalesInfoV2 T1 
					left join Business..TGroupV2 TG on T1.TeamId=TG.TeamId
				) AS TT
				left join Business..TGroupV2 TG1 on TT.TeamId=TG1.TeamId
				left join Business..TGroupV2 TG2 on 
				TG1.[Path] LIKE ('%\' + convert(varchar(50),TG2.TeamID)) 
				 	or TG1.[Path] like ('%\' + convert(varchar(100),TG2.TeamID) + '\%') 
				    or TG1.[Path] like (convert(varchar(50),TG2.TeamID) + '\%') 
				    or TG1.[Path] = convert(varchar(50),TG2.TeamID) 
				LEFT JOIN 
					(select PeopleId,SUM(Amount) as Amount from Business..TPeopleSalesDetailInfoV2 group by PeopleId)
				as BB on TT.PeopleId=BB.PeopleId
		) A	 GROUP by  ParentTeamID
) as AA LEFT JOIN Business..TGroupV2 TB on TB.TeamID=AA.ParentTeamID
UNION 
--以員工為單位獲取各個銷售人員的銷售金額
select TB.TreeNodeNo,TB.TeamID,TB.TeamName,SUM(TP.Amount) as Amount,TP.PeopleId,TPS.PeopleName from Business..TPeopleSalesDetailInfoV2 TP
LEFT JOIN Business..TPeopleSalesInfoV2 TPS on  TPS.PeopleId=TP.PeopleId
LEFT JOIN Business..TGroupV2 TB on TB.TeamID=TPS.TeamID
group by TB.TreeNodeNo,TB.TeamID,TB.TeamName,TP.PeopleId,TPS.PeopleName
ORDER BY TreeNodeNo,PeopleId ASC 

編輯

 

2.3、思路說明

編輯

 

 

3、總結

        隨著數據量增加一些老的sql查詢性能太慢了,經常出現這種查詢超時問題。

編輯

 造成這種問題的原因有很多,一種是sql寫的太爛了,業務層有迴圈查詢。就像我方法一中的那種思想,不可避免你要迴圈查詢出每個團隊的金額再一級一級向上彙總。還有就是不合理的許可權控制。比如你要查詢團隊的銷售金額。因為團隊的關係是一個樹狀結構嘛。假如你是東區的領導,你只能查詢東區及其下所有子團隊的數據,但在許可權判斷這塊,其實是會東區下每個子團隊,以及子團隊的子團隊.....都要判斷一遍你有沒有查詢的許可權。這樣就增加了不必要的負擔。不過這個是歷史遺留問題,是因為之前的許可權結構設計就不完善,也不太好改。

解決方法嘛,目前我就是通過存儲過程取代select查詢,因為存儲過程是預編譯的,所以執行起來開銷比較小所以速度比較快。可以看下這篇詳細瞭解下:為什麼存儲過程比sql語句效率高? - herizai - 博客園 (cnblogs.com)

因為原先的select查詢關聯了好多表以及視圖,各種join的,可讀性很差。我所要做的就是理清這些join之間的關係, 存儲過程中用幾個臨時表把大的join拆成合併成小的join。再加一些註釋什麼的,雖然業務沒有變,只是代碼更容易理解了。速度確實快了一些,不在出現查詢的超時的問題了。

 

 

4、參考資料

為什麼要用存儲過程,存儲過程的優缺點。。_jokeylin的博客-CSDN博客

為什麼存儲過程比sql語句效率高? - herizai - 博客園 (cnblogs.com)


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

-Advertisement-
Play Games
更多相關文章
  • 1. C++常量表達式 constexpr 是 C++ 11 標準新引入的關鍵字,在學習其具體用法和功能之前,我們需要先搞清楚 C++ 常量表達式的含義。 所謂常量表達式,指的就是由多個(≥1)常量組成的表達式。換句話說,如果表達式中的成員都是常量,那麼該表達式就是一個常量表達式。這也意味著,常量表 ...
  • 眾所周知,非同步併發編程可以幫助程式更好地處理阻塞操作,比如網路 IO 操作或文件 IO 操作,避免因等待這些操作完成而導致程式卡住的情況。雲存儲文件傳輸場景正好包含網路 IO 操作和文件 IO 操作,比如業內相對著名的七牛雲存儲,官方sdk的預設阻塞傳輸模式雖然差強人意,但未免有些循規蹈矩,不夠銳意 ...
  • Docker-compose是什麼?能幹嘛?解決了哪些痛點? 是什麼? Docker-compose是Docker官方推出 的一個工具軟體,可以管理多個Docker容器組成的一個應用。你需要編寫一個一個YAML格式的配置文件:docker-compose.yml。寫好多個容器之間的調用關係。然後,只 ...
  • 前言 今天給大家介紹的是Python爬蟲批量下載音樂飆升榜並保存本地,在這裡給需要的小伙伴們代碼,並且給出一點小心得。 首先是爬取之前應該儘可能偽裝成瀏覽器而不被識別出來是爬蟲,基本的是加請求頭,但是這樣的純文本數據爬取的人會很多,所以我們需要考慮更換代理IP和隨機更換請求頭的方式來對音樂飆升榜進行 ...
  • 數據類型 | 類型 | 用法 | | : : | : : | | BYTE | 8位無符號整數,B代表位元組 | | SBYTE | 8位有符號整數,S代表有符號 | | WORD | 16位無符號整數 | | SWORD | 16位有符號整數 | | DWORD | 32位無符號整數,D代表雙(字) ...
  • 故障說明 MAUI項目是日常使用的項目,一直都好好的 某一天修改了幾行代碼後,突然項目無法編譯了,提示NU1105錯誤 從Git重新拉取一份之前的代碼編譯也是同樣的錯誤,經過半天的查閱,嘗試了幾種方案都沒有效果 刪除obj/project.assets.json 刪除obj 刪除.vs 修複Visu ...
  • 設置USB啟動 當前環境使用的樹莓派版本為:Raspberry Pi 3B,並且已經在SD卡中燒錄系統; 1.使用SD卡燒錄Raspberry Pi OS。 可以只使用Raspberry Pi OS Lite,無桌面環境; 2.Raspberry Pi啟動進入操作系統中後,更新系統環境: sudo ...
  • 一:背景 1. 講故事 相信有很多朋友在學習 SQLSERVER 的時候都聽說過這句話,但大多都是記憶為主,最近在研究 SQLSERVER,所以我們從 底層存儲 的角度來深入理解下。 二:理解數據頁 1. 數據頁的組織 在前面的文章中我也說過,一個 數據頁 是 8k 大小,那這 8k 是如何組織的呢 ...
一周排行
    -Advertisement-
    Play Games
  • 基於.NET Framework 4.8 開發的深度學習模型部署測試平臺,提供了YOLO框架的主流系列模型,包括YOLOv8~v9,以及其系列下的Det、Seg、Pose、Obb、Cls等應用場景,同時支持圖像與視頻檢測。模型部署引擎使用的是OpenVINO™、TensorRT、ONNX runti... ...
  • 十年沉澱,重啟開發之路 十年前,我沉浸在開發的海洋中,每日與代碼為伍,與演算法共舞。那時的我,滿懷激情,對技術的追求近乎狂熱。然而,隨著歲月的流逝,生活的忙碌逐漸占據了我的大部分時間,讓我無暇顧及技術的沉澱與積累。 十年間,我經歷了職業生涯的起伏和變遷。從初出茅廬的菜鳥到逐漸嶄露頭角的開發者,我見證了 ...
  • C# 是一種簡單、現代、面向對象和類型安全的編程語言。.NET 是由 Microsoft 創建的開發平臺,平臺包含了語言規範、工具、運行,支持開發各種應用,如Web、移動、桌面等。.NET框架有多個實現,如.NET Framework、.NET Core(及後續的.NET 5+版本),以及社區版本M... ...
  • 前言 本文介紹瞭如何使用三菱提供的MX Component插件實現對三菱PLC軟元件數據的讀寫,記錄了使用電腦模擬,模擬PLC,直至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1. PLC開發編程環境GX Works2,GX Works2下載鏈接 https:// ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • 1、jQuery介紹 jQuery是什麼 jQuery是一個快速、簡潔的JavaScript框架,是繼Prototype之後又一個優秀的JavaScript代碼庫(或JavaScript框架)。jQuery設計的宗旨是“write Less,Do More”,即倡導寫更少的代碼,做更多的事情。它封裝 ...
  • 前言 之前的文章把js引擎(aardio封裝庫) 微軟開源的js引擎(ChakraCore))寫好了,這篇文章整點js代碼來測一下bug。測試網站:https://fanyi.youdao.com/index.html#/ 逆向思路 逆向思路可以看有道翻譯js逆向(MD5加密,AES加密)附完整源碼 ...
  • 引言 現代的操作系統(Windows,Linux,Mac OS)等都可以同時打開多個軟體(任務),這些軟體在我們的感知上是同時運行的,例如我們可以一邊瀏覽網頁,一邊聽音樂。而CPU執行代碼同一時間只能執行一條,但即使我們的電腦是單核CPU也可以同時運行多個任務,如下圖所示,這是因為我們的 CPU 的 ...
  • 掌握使用Python進行文本英文統計的基本方法,並瞭解如何進一步優化和擴展這些方法,以應對更複雜的文本分析任務。 ...
  • 背景 Redis多數據源常見的場景: 分區數據處理:當數據量增長時,單個Redis實例可能無法處理所有的數據。通過使用多個Redis數據源,可以將數據分區存儲在不同的實例中,使得數據處理更加高效。 多租戶應用程式:對於多租戶應用程式,每個租戶可以擁有自己的Redis數據源,以確保數據隔離和安全性。 ...