SQLServer學習筆記系列6


一.寫在前面的話

時間是我們每個人都特別熟悉的,但是到底它是什么,用什么來衡量,可能很多人會愣在那里。時間可以見證一切,也可以消磨一切,那些過往的點點滴滴可思可憶。回想往年清明節過后,在家鄉的晚上總能聽見陣陣的青蛙叫聲,那是清脆的叫聲,那是家鄉的味道。時間一轉眼,貌似那些日子已離我遠去好久,在城市的喧囂浮華中,找尋不到那種內心的寧靜。感嘆時間流逝的同時,懷念過去的點點滴滴。我想在繁華的都市中尋找一種安定的心情來學習,或許是一種不錯的方式。學習才會讓我們認清自己,找回自我,做內心的強者,不驕不躁,積極進取,陽光自信。努力的人最幸運!Just do it!

                                             從今以后,別再過你應該過的人生,去過你想過的人生吧!——梭羅

二.派生表

派生表也就是特殊的子查詢,用在from之后的子查詢,同時這里注意一點:派生表必須起別名。

例如:加入我們要查詢,顧客信息,以及每個國家所擁有的顧客數量。那么我們可以這樣寫:

 

1  
2  SELECT * FROM
3  (
4  SELECT custid,COUNT(*) OVER(PARTITION BY country) AS N'顧客數量'
5  FROM Sales.Customers
6  ) t

括號里面的子句就相當於一個派生表,假如我們在這里不給 COUNT(*) OVER(PARTITION BY country) 起別名,那么select * 就不知道取出哪一列,所以會報錯。

同時有另一種起別名的方法,也就是在外面其別名,同樣可以達到效果,寫法如下:

1  SELECT * FROM
2  (
3  SELECT custid,COUNT(*) OVER(PARTITION BY country)
4  FROM Sales.Customers
5  ) t(custid,顧客數量)

三.CTE(公用表表達式)

(1)常見CTE用法

如何使用CTE,在這里必須先定義,並且命名。定義形式如下:

1  WITH USE_Customers
2  AS
3  (
4    SELECT companyname,country
5    FROM Sales.Customers 
6    WHERE country='USA'
7  )

那么我們要使用CTE,只要查詢對應的CTE名稱即可。

1  WITH USE_Customers
2  AS
3  (
4    SELECT companyname,country
5    FROM Sales.Customers 
6    WHERE country='USA'
7  )
8  
9  SELECT * FROM USE_Customers

結果如圖所示:

但是要注意的一點是:使用CTE時,外部查詢一旦完成,那么CTE的生命期就結束了,如果需要在查詢這個CTE,此時失效。

同時也可以給CTE每列其別名,其中也存在在表達式內部和外部起別名,看個人喜歡哪種方式。

1  WITH USE_Customers
2  AS
3  (
4    SELECT companyname AS N'公司名',country AS N'國家名'
5    FROM Sales.Customers 
6    WHERE country='USA'
7  )
8  
9  SELECT * FROM USE_Customers
1  WITH USE_Customers(公司名,國家名)
2  AS
3  (
4    SELECT companyname ,country 
5    FROM Sales.Customers 
6    WHERE country='USA'
7  )
8  
9  SELECT * FROM USE_Customers

其查詢結果都為:

(2)有參數的CTE

使用有參數的CTE可以多次使用,只需要改變參數的傳遞值即可。

例如:查詢每個國家的顧客信息,國家存在USA,UK........,多個國家,此時我們將國家作為一個變量,類似於C#中的參數。聲明CTE如下:

 1  DECLARE @country NVARCHAR(300);
 2  SET @country='USA';
 3  WITH USE_Customers(公司名,國家名)
 4  AS
 5  (
 6    SELECT companyname ,country 
 7    FROM Sales.Customers 
 8    WHERE country=@country
 9  )
10  
11  SELECT * FROM USE_Customers

假如要查詢UK國家的一些顧客信息,那么只需要改變set @country=‘UK’就可以查詢了,不需要改變CTE的結構。

 1  DECLARE @country NVARCHAR(300);
 2  SET @country='UK';
 3  WITH USE_Customers(公司名,國家名)
 4  AS
 5  (
 6    SELECT companyname ,country 
 7    FROM Sales.Customers 
 8    WHERE country=@country
 9  )
10  
11  SELECT * FROM USE_Customers

(3)復雜的CTE使用

【1】在這里,假如我們要查詢年度訂單數量在10以上的顧客信息。按照一般的思維,我們可以講訂單按照年度分組,然后用聚合函數count統計每個顧客下的訂單數量,接着找出大於訂單數量大於10的顧客信息。所以我們可以下sql如下:

查詢結果如圖所示:

【2】那么同時可以使用子查詢,即派生表依次來出來問題。

第一步,先查詢訂單的年份,顧客的信息

1  SELECT  YEAR(orderdate) AS orderyear,custid
2  FROM Sales.Orders

第二步,根據年份進行分組,統計每位顧客的訂單數量。第一步查詢出來的結果作為子查詢,即派生表。

1  SELECT orderyear,custid,COUNT(*) AS N'訂單數量'
2  FROM
3  (
4      SELECT  YEAR(orderdate) AS orderyear,custid 
5      FROM Sales.Orders
6  ) AS t1
7  GROUP BY orderyear,custid

 

第三步,找出訂單數量大於10的顧客信息。

 1  SELECT orderyear,custid,ordercount
 2  FROM
 3  (
 4  SELECT orderyear,custid,COUNT(*) AS  ordercount
 5  FROM
 6          (
 7              SELECT  YEAR(orderdate) AS orderyear,custid 
 8              FROM Sales.Orders
 9          ) AS t1
10          GROUP BY orderyear,custid
11  ) AS t2
12  WHERE ordercount >10

是不是感覺使用子查詢,一層套一層,不過分析以后,符合我們一般解決問題的思路先做哪一步,再做哪一步。在這里我們可以使用新學的CTE來解決問題。

【3】使用CTE來解決年度訂單數量在10以上的顧客信息。

好處就是定義好的CTE,可以直接拿來作為一個派生表使用,那么我們可以這樣定義CTE,按照上述的分析,分別定義三個CTE,最后查詢出結果:

需要注意的一點事:定義多個CTE之間用逗號(,)隔開即可,不需要使用with重新定。

 1  WITH yearorder01
 2  AS
 3  (
 4        SELECT YEAR(orderdate) AS orderyear,custid
 5        FROM Sales.Orders
 6  ),
 7  yearorder02
 8  AS 
 9  (
10          SELECT orderyear,custid,COUNT(*) AS ordercount
11          FROM yearorder01
12          GROUP BY orderyear,custid
13  ),
14  yearorder03
15  AS
16  (
17         SELECT orderyear,custid,ordercount
18         FROM yearorder02
19         WHERE  ordercount>10
20  )
21  
22  
23  SELECT * FROM yearorder03

查詢結果如圖所示,跟子查詢結果相同:

四.多CTE

多個CTE的使用會讓我們更加方便的進行相關子查詢。例如:假如我們要查詢年度有多少顧客,測試該怎么處理了?我們會想到,還是根據年度分組,然后聚合count(custid)求出每年度有多少顧客,其中要注意去重,即相同的custid屬於同一個顧客,所以要寫成count(distinct custid)。sql如下:

1  
2  SELECT YEAR(orderdate) AS orderyear,COUNT(DISTINCT custid) AS  custcount
3  FROM Sales.Orders
4  GROUP BY YEAR(orderdate)

現在加入有一個需求,就是要做一個查詢,求出當年的顧客數量以及前一年的顧客數量,同時算出兩者之間顧客數量隔了多少,我們又該怎么處理了?要求前一年和當前年的差,那么就可以想到,是否可以把當前查出來的結果集做一次連接,不就可以得到前一年和當前年的嘛,所以sql如下:

 1  SELECT pre_orderyear,now_orderyear,pre_custcount,now_custcount,
 2        (now_custcount-pre_custcount) AS N'顧客數量差'
 3  FROM 
 4  (
 5  SELECT YEAR(orderdate) AS now_orderyear,COUNT(DISTINCT custid) AS now_custcount 
 6  FROM Sales.Orders
 7  GROUP BY YEAR(orderdate)
 8  ) AS t1
 9  LEFT JOIN 
10  (
11   SELECT YEAR(orderdate) AS pre_orderyear,COUNT(DISTINCT custid) AS  pre_custcount
12  FROM Sales.Orders
13  GROUP BY YEAR(orderdate)
14  ) AS t2
15  ON t1.now_orderyear=t2.pre_orderyear+1;

其中就是將結果集做一個自身的連接,但是我們是不是發現這樣寫顯得很冗余了,作為程序猿的我們就是要寫出盡量簡潔明了的code,Don‘t  repeat yourself !在這里我們的CTE就發揮出他的作用,定義的CTE可以在下一步中使用,作為查詢條件。那么我們看看用CTE如何實現?同理,我們使用CTE之前就必須先定義CTE,這里我們定義一個CTE,查詢出每年的顧客數量,其實跟上面的sql一樣,只不過定義成CTE的形式。

 1  WITH custcount
 2  AS
 3  (
 4  SELECT YEAR(orderdate) AS orderyear,COUNT(DISTINCT custid) AS  custcount
 5  FROM Sales.Orders
 6  GROUP BY YEAR(orderdate)
 7  )
 8  
 9  SELECT t1.orderyear AS nowYear,t2.orderyear AS preYear,t1.custcount AS nowcount,t2.custcount AS precount,
10        (t1.custcount-t2.custcount) AS N'顧客數量差'
11  FROM custcount t1
12  LEFT JOIN  custcount t2
13  ON t1.orderyear=t2.orderyear+1;

其結果如圖所示:

其實我們從sql語句可以看出,CTE的簡潔之處,就在於,可以將已定義的CTE重復使用進行連接,相比於子查詢自身的連接,更加簡潔,容易操作,加入需要的計算,我們只需要改變CTE的定義即可。

五.遞歸CTE

遞歸的CTE在遞歸查找中應用的非常多,其實可以看做一種層次的查找關系,比如公司一般情況下是分一級部門,二級部門,三級部門.......那么一級部門下包含多個二級部門,二級部門又包含多個三級部門等等.....,舉例說明:公司內部有上下級的關系,假如我們要查找某個員工的上級是誰,一般情況下我們可以做一次自身的連接,找出上級是誰。

例如:查找所有顧客的上級信息。也就是在HR.employees里面的mgrid等於員工的empid,那么找到的那個員工幾位mgrid的上司。sql如下:

1  SELECT t1.empid,t1.mgrid,t1.lastname,t2.empid,t2.lastname
2  FROM HR.Employees t1 LEFT JOIN hr.Employees t2 
3  ON t1.mgrid =t2.empid

因為為了確保所有的員工信息都顯示,所以在這里用到餓左連接,保證左邊表里面的員工都存在。其結果如圖:

這里再進一步討論,加入我們要查詢某個員工下面所有的下屬信息,怎么辦了?比如要查詢2號員工的下屬信息,我們這可以這樣寫sql:

1  SELECT * FROM 
2  hr.Employees WHERE mgrid=2

我們了可以看到2號員工下屬有3號和5號,那么5號員工可能也有下屬,接着查詢:

1  SELECT * FROM HR.Employees
2  WHERE mgrid in
3  (SELECT empid FROM 
4  hr.Employees WHERE mgrid=2
5  ) 

我們可以看到,查出的結果中有4,6,7,8,9,那么4,6,7,8,9下面是否還有下屬了,是不是還需要接着查詢了,這樣查下去就是無止盡的,因為我們不知道有多少層,所以遞歸CTE就是解決這樣的問題的,遞歸CTE可以查詢出所以層級的情況,直至結束。首先遞歸CTE需要定義一個起點,即查詢起點。然后根據定義的CTE自身做遞歸查詢,直至找出所有的節點信息。如下:

 1 DECLARE @mgrid INT;
 2 SET @mgrid=2;
 3 WITH Emplist
 4 AS
 5 (
 6     --此處為起點,執行一次
 7     SELECT empid,lastname,mgrid
 8     FROM HR.Employees
 9     WHERE mgrid=@mgrid
10     UNION ALL
11     
12     --遞歸開始
13     
14     SELECT e.empid,e.lastname,e.mgrid
15     FROM HR.Employees e INNER JOIN Emplist m
16     ON e.mgrid=m.empid 
17     
18 )
19 
20 SELECT * FROM Emplist

定義一個變量,接受要查詢的員工id,設為2,即查詢2號員工下面所有的下屬信息。

關於CTE的運用還有很多,這里只列出一些基本的用法,如有想法,可以提出,希望一起學習!

sqlserver學習筆記1:http://www.cnblogs.com/liupeng61624/p/4354983.html

sqlserver學習筆記2:http://www.cnblogs.com/liupeng61624/p/4367580.html

sqlserver學習筆記3:http://www.cnblogs.com/liupeng61624/p/4375135.html

sqlserver學習筆記4:http://www.cnblogs.com/liupeng61624/p/4388959.html

sqlserver學習筆記5:http://www.cnblogs.com/liupeng61624/p/4392746.html

 

希望各位大牛給出指導,不當之處虛心接受學習!謝謝!

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM