適用於:SQL Server
Azure SQL 資料庫
Azure SQL 受控執行個體
Azure Synapse Analytics
分析平台系統(PDW)
Microsoft Fabric 中的 SQL 分析端點
Microsoft Fabric 中的倉儲
Microsoft Fabric 中的 SQL 資料庫
指定稱為通用資料表運算式 (CTE) 的暫存具名結果集。 這是衍生自簡單的查詢,並定義於單SELECT一 、INSERT、、 UPDATEMERGE或 DELETE 語句的執行範圍內。 這個子句也可以在語句中 CREATE VIEW 當做其定義 SELECT 語句的一部分使用。 通用資料表運算式可以包括指向本身的參考。 這稱為遞迴通用資料表運算式。
如需詳細資訊,請參閱 使用通用數據表表達式的遞歸查詢。
Syntax
[ WITH <common_table_expression> [ , ...n ] ]
<common_table_expression>::=
expression_name [ ( column_name [ , ...n ] ) ]
AS
( CTE_query_definition )
Arguments
expression_name
通用資料表運算式的有效識別碼。
expression_name 必須與相同 WITH <common_table_expression> 子句中定義的任何其他通用數據表運算式名稱不同,但 expression_name 與基表或檢視表的名稱相同。 查詢中 expression_name 的任何參考都會使用通用數據表運算式,而不是基底物件。
column_name
在一般資料表運算式中,指定資料行名稱。 在單一 CTE 定義內,名稱不可重複。 指定的數據行名稱數目必須符合 CTE_query_definition結果集中的數據行數目。 只有在查詢定義提供了所有結果資料行的個別名稱時,資料行名稱清單才是選擇性的。
CTE_query_definition
指定 SELECT 結果集填入通用數據表表達式的語句。
SELECT
CTE_query_definition的語句必須符合與建立檢視相同的需求,但 CTE 無法定義另一個 CTE。 如需詳細資訊,請參閱一節和 CREATE VIEW。
如果定義了一個以上的 CTE_query_definition ,查詢定義必須由下列其中一個 set 運算子聯結: UNION ALL、 UNION、 EXCEPT或 INTERSECT。
使用方針
通用數據表表達式的查詢結果不會具體化。 具名結果集的每個外部參考都需要重新執行已定義的查詢。 對於需要多個具名結果集參考的查詢,請考慮改用 暫存物件 。
您無法在通用資料表運算式中執行預存程式。
如需遞歸和非遞歸 CTE 的使用指導方針,請參閱下列各節。
非遞歸通用數據表表達式的指導方針
Note
下列方針適用於非遞迴的通用資料表運算式。 如需適用於遞歸通用數據表表達式的指導方針,請參閱 遞歸通用數據表表達式的指導方針。
CTE 後面必須接著參考部分或所有 CTE 數據行的單SELECT一 、INSERT、、 UPDATEMERGE或 DELETE 語句。 您也可以在 CREATE VIEW 陳述式中,將 CTE 指定為檢視之定義 SELECT 陳述式的一部分。
您可以在非遞迴的 CTE 中,定義多個 CTE 查詢定義。 這些定義必須透過下列其中一個設定運算子來組合:UNION ALL、UNION、INTERSECT 或 EXCEPT。
CTE 可以參考其本身,以及先前在相同 WITH 子句中所定義的 CTE。 不允許轉送參考。
不允許在 CTE 中指定多個 WITH 子句。 例如,如果 CTE_query_definition 包含子查詢,該子查詢就無法包含定義另一個 CTE 的巢狀 WITH 子句。
如需 Microsoft Fabric 中巢狀 CTE 的詳細資訊,請參閱 網狀架構數據倉儲中巢狀通用數據表運算式 (CTE)。
下列子句無法在 CTE_query_definition中使用:
-
ORDER BY(除非TOP指定 或OFFSET/FETCH子句) INTO-
OPTION具有查詢提示的 子句 1 FOR BROWSE
1 子 OPTION 句無法在 CTE 定義內使用。 它只能在最 SELECT 外層的語句中使用。
當批次中的陳述式使用 CTE 時,在 CTE 之前的陳述式,後面必須接著分號。
參考 CTE 的查詢可用來定義資料指標。
在 CTE 中,可以參考遠端伺服器的資料表。
執行 CTE 時,任何參考 CTE 的提示都可能會與其他在 CTE 存取其基礎表時所探索到的提示衝突,與參考查詢中檢視的提示相同。 當這種情況發生時,查詢會傳回錯誤:
遞歸通用數據表表達式的指導方針
Note
下列指導方針適用於定義遞歸通用數據表運算式。 如需適用於非遞歸 CTE 的指導方針,請參閱 非遞歸通用數據表表達式的指導方針。
遞歸 CTE 定義必須包含至少兩個 CTE 查詢定義:錨點成員和遞歸成員。 您可以定義多個錨點成員和遞迴成員;不過,所有錨點成員查詢定義都必須放在第一個遞迴成員定義的前面。 除非 CTE 查詢定義參考 CTE 本身,否則,它們都是錨點成員。
錨點成員必須結合下列其中一個集合運算符: UNION ALL、 UNION、 INTERSECT或 EXCEPT。
UNION ALL 是最後一個錨點成員與第一個遞歸成員之間唯一允許的 set 運算符,在結合多個遞歸成員時。
錨點和遞迴成員中的資料行數目必須相同。
遞迴成員資料行的資料類型必須與錨點成員中對應資料行的資料類型相同。
FROM遞歸成員的 子句只能參考 CTE expression_name一次。
遞歸成員 CTE_query_definition 中不允許下列專案:
SELECT DISTINCTGROUP BY-
PIVOT1 HAVING- 純量聚合
TOP-
LEFT、RIGHT、OUTER JOIN(允許使用INNER JOIN) - Subqueries
- 套用至 CTE_query_definition內 CTE 遞歸參考的提示。
1 當資料庫相容性層級為 110 或更高版本時。 請參閱 SQL Server 2016 中 Database Engine 功能的重大變更。
下列方針適用於使用遞迴通用資料表運算式:
遞迴 CTE 能夠傳回的所有資料行都可為 Null,不論參與的
SELECT陳述式所傳回之資料行 Null 屬性為何,都是如此。撰寫不正確的遞歸 CTE 可能會導致無限迴圈。 例如,若遞迴成員查詢定義對父資料行與子資料行傳回相同的值,就會建立無限迴圈。 若要防止無限迴圈,您可以使用 、 、 或
MAXRECURSION語句 子句032767OPTION中的INSERThint和值UPDATEDELETE,限制特定語句SELECT允許的遞歸層級數目。 這可讓您控制陳述式的執行,直到產生迴圈的程式碼問題解決為止。 伺服器範圍的預設值是 100。 當指定 0 時,不會套用任何限制。 每個陳述式只能指定一個MAXRECURSION值。 如需詳細資訊,請參閱 查詢提示。包含遞迴通用資料表運算式的檢視無法用來更新資料。
您可以使用 CTE 在查詢上定義資料指標。 CTE 是定義數據指標結果集 的 select_statement 自變數。 遞迴 CTE 只能使用僅限向前快轉和靜態 (快照集) 資料指標。 如果在遞迴 CTE 中指定了另一種資料指標類型,就會將資料指標類型轉換成靜態。
在 CTE 中,可以參考遠端伺服器的資料表。 如果在 CTE 遞迴成員參考遠端伺服器,便會為每個遠端資料表各建立一項多工緩衝處理,以便在本機重複存取資料表。 如果是 CTE 查詢,索引多任務緩衝處理/延遲多任務緩衝處理會顯示在查詢計劃中,而且會有額外的
WITH STACK述詞。 這是確認適當遞迴的一種方式。CTE 遞迴部分中的分析和彙總函式會套用至目前遞迴層級的集合,而不會套用至 CTE 的集合。
ROW_NUMBER之類的函式只會針對目前遞迴層級傳遞給它們的資料子集運作,而不會針對傳遞給 CTE 遞迴部分的整個資料集運作。 如需詳細資訊,請參閱後續的範例「I. 在遞迴 CTE 中使用分析函式」。
Azure Synapse Analytics and Analytics Platform System 中的常見數據表表達式 (PDW)
Azure Synapse Analytics and Analytics Platform System 中目前 CTE 的實作具有下列功能和需求:
可以在
SELECT陳述式中指定 CTE。可以在
CREATE VIEW陳述式中指定 CTE。可以在
CREATE TABLE AS SELECT(CTAS) 陳述式中指定 CTE。可以在
CREATE REMOTE TABLE AS SELECT(CRTAS) 陳述式中指定 CTE。可以在
CREATE EXTERNAL TABLE AS SELECT(CETAS) 陳述式中指定 CTE。可以從 CTE 參考遠端資料表。
可以從 CTE 參考外部資料表。
可以在 CTE 中定義多個 CTE 查詢定義。
CTE 後面可以接著
SELECT、、INSERTUPDATE、DELETE、 或MERGE語句。不支援包含自我參考的通用資料表運算式 (遞迴通用資料表運算式)。
不允許在 CTE 中指定多個
WITH子句。 例如,如果 CTE 查詢定義包含子查詢,該子查詢就不能包含定義另一個 CTE 的巢狀WITH子句。ORDER BY子句不能用於CTE_query_definition,除非指定 子句。TOP當批次中的陳述式使用 CTE 時,在 CTE 之前的陳述式,後面必須接著分號。
在備妥的
sp_prepare語句中使用 時,CTE 的行為方式與SELECTAPS PDW 中的其他語句相同。 不過,如果使用 CETAS 做為 所sp_prepare準備 CETAS 的一部分,則行為可能會因為 針對sp_prepare實作系結的方式而延遲 SQL Server 和其他 APS PDW 語句。 如果SELECT參考 CTE 使用 CTE 中不存在的錯誤資料行,則sp_prepare傳遞時不會偵測到錯誤,但在期間sp_execute擲回錯誤。
Examples
A. 建立通用資料表運算式
下列範例顯示 Adventure Works Cycles 中每個銷售代表每年的銷售訂單總數。
-- Define the CTE expression name and column list.
WITH Sales_CTE (SalesPersonID, SalesOrderID, SalesYear)
AS
-- Define the CTE query.
(
SELECT SalesPersonID, SalesOrderID, YEAR(OrderDate) AS SalesYear
FROM Sales.SalesOrderHeader
WHERE SalesPersonID IS NOT NULL
)
-- Define the outer query referencing the CTE name.
SELECT SalesPersonID,
COUNT(SalesOrderID) AS TotalSales,
SalesYear
FROM Sales_CTE
GROUP BY SalesYear, SalesPersonID
ORDER BY SalesPersonID, SalesYear;
B. 使用通用資料表運算式來限制計數和報告平均值
下列範例顯示業務人員所有年度的平均銷售訂單數目。
WITH Sales_CTE (SalesPersonID, NumberOfOrders)
AS
(
SELECT SalesPersonID, COUNT(*)
FROM Sales.SalesOrderHeader
WHERE SalesPersonID IS NOT NULL
GROUP BY SalesPersonID
)
SELECT AVG(NumberOfOrders) AS "Average Sales Per Person"
FROM Sales_CTE;
C. 在單一查詢中使用多個 CTE 定義
下列範例會示範如何在單一查詢中定義一個以上的 CTE。 逗號可用來分隔 CTE 查詢定義。 函 FORMAT 式用來以貨幣格式顯示貨幣金額,是在 SQL Server 2012 (11.x) 中引進的。
WITH Sales_CTE (SalesPersonID, TotalSales, SalesYear)
AS
-- Define the first CTE query.
(
SELECT SalesPersonID,
SUM(TotalDue) AS TotalSales,
YEAR(OrderDate) AS SalesYear
FROM Sales.SalesOrderHeader
WHERE SalesPersonID IS NOT NULL
GROUP BY SalesPersonID, YEAR(OrderDate)
), -- Use a comma to separate multiple CTE definitions.
-- Define the second CTE query, which returns sales quota data by year for each sales person.
Sales_Quota_CTE (BusinessEntityID, SalesQuota, SalesQuotaYear)
AS
(
SELECT BusinessEntityID,
SUM(SalesQuota) AS SalesQuota,
YEAR(QuotaDate) AS SalesQuotaYear
FROM Sales.SalesPersonQuotaHistory
GROUP BY BusinessEntityID, YEAR(QuotaDate)
)
-- Define the outer query by referencing columns from both CTEs.
SELECT SalesPersonID,
SalesYear,
FORMAT(TotalSales, 'C', 'en-us') AS TotalSales,
SalesQuotaYear,
FORMAT(SalesQuota, 'C', 'en-us') AS SalesQuota,
FORMAT(TotalSales - SalesQuota, 'C', 'en-us') AS Amt_Above_or_Below_Quota
FROM Sales_CTE
INNER JOIN Sales_Quota_CTE
ON Sales_Quota_CTE.BusinessEntityID = Sales_CTE.SalesPersonID
AND Sales_CTE.SalesYear = Sales_Quota_CTE.SalesQuotaYear
ORDER BY SalesPersonID, SalesYear;
以下為部分結果集。
SalesPersonID SalesYear TotalSales SalesQuotaYear SalesQuota Amt_Above_or_Below_Quota
------------- --------- ----------- -------------- ---------- ----------------------------------
274 2005 $32,567.92 2005 $35,000.00 ($2,432.08)
274 2006 $406,620.07 2006 $455,000.00 ($48,379.93)
274 2007 $515,622.91 2007 $544,000.00 ($28,377.09)
274 2008 $281,123.55 2008 $271,000.00 $10,123.55
D. 使用遞迴通用資料表運算式顯示多層級的遞迴
下列範例會顯示經理及向經理提出報告的員工的階層式清單。 此範例會開始建立並擴展 dbo.MyEmployees 資料表。
-- Create an Employee table.
CREATE TABLE dbo.MyEmployees
(
EmployeeID SMALLINT NOT NULL,
FirstName NVARCHAR (30) NOT NULL,
LastName NVARCHAR (40) NOT NULL,
Title NVARCHAR (50) NOT NULL,
DeptID SMALLINT NOT NULL,
ManagerID SMALLINT NULL,
CONSTRAINT PK_EmployeeID PRIMARY KEY CLUSTERED (EmployeeID ASC),
CONSTRAINT FK_MyEmployees_ManagerID_EmployeeID FOREIGN KEY (ManagerID) REFERENCES dbo.MyEmployees (EmployeeID)
);
-- Populate the table with values.
INSERT INTO dbo.MyEmployees VALUES
(1, N'Ken', N'Sánchez', N'Chief Executive Officer', 16, NULL),
(273, N'Brian', N'Welcker', N'Vice President of Sales', 3, 1),
(274, N'Stephen', N'Jiang', N'North American Sales Manager', 3, 273),
(275, N'Michael', N'Blythe', N'Sales Representative', 3, 274),
(276, N'Linda', N'Mitchell', N'Sales Representative', 3, 274),
(285, N'Syed', N'Abbas', N'Pacific Sales Manager', 3, 273),
(286, N'Lynn', N'Tsoflias', N'Sales Representative', 3, 285),
(16, N'David', N'Bradley', N'Marketing Manager', 4, 273),
(23, N'Mary', N'Gibson', N'Marketing Specialist', 4, 16);
WITH DirectReports (ManagerID, EmployeeID, Title, EmployeeLevel) AS
(
SELECT ManagerID, EmployeeID, Title, 0 AS EmployeeLevel
FROM dbo.MyEmployees
WHERE ManagerID IS NULL
UNION ALL
SELECT e.ManagerID, e.EmployeeID, e.Title, EmployeeLevel + 1
FROM dbo.MyEmployees AS e
INNER JOIN DirectReports AS d
ON e.ManagerID = d.EmployeeID
)
SELECT ManagerID, EmployeeID, Title, EmployeeLevel
FROM DirectReports
ORDER BY ManagerID;
使用遞迴通用資料表運算式顯示兩個層級的遞迴
下列範例會顯示經理及向經理提出報告的員工。 傳回的層級數目只限兩個。
WITH DirectReports (ManagerID, EmployeeID, Title, EmployeeLevel) AS
(
SELECT ManagerID, EmployeeID, Title, 0 AS EmployeeLevel
FROM dbo.MyEmployees
WHERE ManagerID IS NULL
UNION ALL
SELECT e.ManagerID, e.EmployeeID, e.Title, EmployeeLevel + 1
FROM dbo.MyEmployees AS e
INNER JOIN DirectReports AS d
ON e.ManagerID = d.EmployeeID
)
SELECT ManagerID, EmployeeID, Title, EmployeeLevel
FROM DirectReports
WHERE EmployeeLevel <= 2;
使用遞迴通用資料表運算式顯示階層式清單
下列範例會新增經理和員工的姓名及其職稱。 各個層級會進行縮排,更明顯地強調經理和員工的階層。
WITH DirectReports (Name, Title, EmployeeID, EmployeeLevel, Sort) AS
(
SELECT CONVERT (VARCHAR (255), e.FirstName + ' ' + e.LastName),
e.Title,
e.EmployeeID,
1,
CONVERT (VARCHAR (255), e.FirstName + ' ' + e.LastName)
FROM dbo.MyEmployees AS e
WHERE e.ManagerID IS NULL
UNION ALL
SELECT CONVERT (VARCHAR (255), REPLICATE('| ', EmployeeLevel) + e.FirstName + ' ' + e.LastName),
e.Title,
e.EmployeeID,
EmployeeLevel + 1,
CONVERT (VARCHAR (255), RTRIM(Sort) + '| ' + FirstName + ' ' + LastName)
FROM dbo.MyEmployees AS e
INNER JOIN DirectReports AS d
ON e.ManagerID = d.EmployeeID
)
SELECT EmployeeID, Name, Title, EmployeeLevel
FROM DirectReports
ORDER BY Sort;
使用 MAXRECURSION 取消陳述式
您可以利用 MAXRECURSION 來防止形式不良的遞迴 CTE 進入無限迴圈。 下列範例會刻意建立無限迴圈,然後利用 MAXRECURSION 提示,將遞迴層級限制為 2。
--Creates an infinite loop
WITH cte (EmployeeID, ManagerID, Title) AS
(
SELECT EmployeeID, ManagerID, Title
FROM dbo.MyEmployees
WHERE ManagerID IS NOT NULL
UNION ALL
SELECT cte.EmployeeID, cte.ManagerID, cte.Title
FROM cte
INNER JOIN dbo.MyEmployees AS e
ON cte.ManagerID = e.EmployeeID
)
--Uses MAXRECURSION to limit the recursive levels to 2
SELECT EmployeeID, ManagerID, Title
FROM cte
OPTION (MAXRECURSION 2);
修正程式代碼錯誤之後,不再需要 MAXRECURSION。 下列範例會顯示更正的程式碼。
WITH cte (EmployeeID, ManagerID, Title) AS
(
SELECT EmployeeID, ManagerID, Title
FROM dbo.MyEmployees
WHERE ManagerID IS NOT NULL
UNION ALL
SELECT e.EmployeeID, e.ManagerID, e.Title
FROM dbo.MyEmployees AS e
INNER JOIN cte
ON e.ManagerID = cte.EmployeeID
)
SELECT EmployeeID, ManagerID, Title
FROM cte;
E. 使用通用資料表運算式,在 SELECT 陳述式中選擇性地逐步執行遞迴關聯性
下列範例會顯示建立 ProductAssemblyID = 800 的自行車時,所需要之產品組件和元件的階層。
USE AdventureWorks2022;
GO
WITH Parts (AssemblyID, ComponentID, PerAssemblyQty, EndDate, ComponentLevel) AS
(
SELECT b.ProductAssemblyID,
b.ComponentID,
b.PerAssemblyQty,
b.EndDate,
0 AS ComponentLevel
FROM Production.BillOfMaterials AS b
WHERE b.ProductAssemblyID = 800
AND b.EndDate IS NULL
UNION ALL
SELECT bom.ProductAssemblyID,
bom.ComponentID,
p.PerAssemblyQty,
bom.EndDate,
ComponentLevel + 1
FROM Production.BillOfMaterials AS bom
INNER JOIN Parts AS p
ON bom.ProductAssemblyID = p.ComponentID
AND bom.EndDate IS NULL
)
SELECT AssemblyID,
ComponentID,
Name,
PerAssemblyQty,
EndDate,
ComponentLevel
FROM Parts AS p
INNER JOIN Production.Product AS pr
ON p.ComponentID = pr.ProductID
ORDER BY ComponentLevel, AssemblyID, ComponentID;
F. 在 UPDATE 陳述式中使用遞迴 CTE
下列範例會 PerAssemblyQty 更新用來建置產品 'Road-550-W Yellow, 44' (ProductAssemblyID 800)的所有元件值。 通用資料表運算式會傳回一份階層式清單,其中包含用來建立 ProductAssemblyID 800 的組件、用來建立這些組件的元件等等。 只會修改通用資料表運算式所傳回的資料列。
USE AdventureWorks2022;
GO
WITH Parts (AssemblyID, ComponentID, PerAssemblyQty, EndDate, ComponentLevel) AS
(
SELECT b.ProductAssemblyID,
b.ComponentID,
b.PerAssemblyQty,
b.EndDate,
0 AS ComponentLevel
FROM Production.BillOfMaterials AS b
WHERE b.ProductAssemblyID = 800
AND b.EndDate IS NULL
UNION ALL
SELECT bom.ProductAssemblyID,
bom.ComponentID,
p.PerAssemblyQty,
bom.EndDate,
ComponentLevel + 1
FROM Production.BillOfMaterials AS bom
INNER JOIN Parts AS p
ON bom.ProductAssemblyID = p.ComponentID
AND bom.EndDate IS NULL
)
UPDATE Production.BillOfMaterials
SET PerAssemblyQty = c.PerAssemblyQty * 2
FROM Production.BillOfMaterials AS c
INNER JOIN Parts AS d
ON c.ProductAssemblyID = d.AssemblyID
WHERE d.ComponentLevel = 0;
H. 使用多個錨點和遞迴成員
下列範例會利用多個錨點和遞迴成員來傳回指定人員的所有上階。 它會建立一份資料表,且會插入值來建立遞迴 CTE 所傳回的家族族譜。
-- Genealogy table
IF OBJECT_ID('dbo.Person', 'U') IS NOT NULL
DROP TABLE dbo.Person;
GO
CREATE TABLE dbo.Person
(
ID INT,
Name VARCHAR (30),
Mother INT,
Father INT
);
GO
INSERT dbo.Person VALUES
(1, 'Sue', NULL, NULL),
(2, 'Ed', NULL, NULL),
(3, 'Emma', 1, 2),
(4, 'Jack', 1, 2),
(5, 'Jane', NULL, NULL),
(6, 'Bonnie', 5, 4),
(7, 'Bill', 5, 4);
GO
-- Create the recursive CTE to find all of Bonnie's ancestors.
WITH Generation (ID) AS
(
-- First anchor member returns Bonnie's mother.
SELECT Mother
FROM dbo.Person
WHERE Name = 'Bonnie'
UNION
-- Second anchor member returns Bonnie's father.
SELECT Father
FROM dbo.Person
WHERE Name = 'Bonnie'
UNION ALL
-- First recursive member returns male ancestors of the previous generation.
SELECT Person.Father
FROM Generation, Person
WHERE Generation.ID = Person.ID
UNION ALL
-- Second recursive member returns female ancestors of the previous generation.
SELECT Person.Mother
FROM Generation, dbo.Person
WHERE Generation.ID = Person.ID
)
SELECT Person.ID, Person.Name, Person.Mother, Person.Father
FROM Generation, dbo.Person
WHERE Generation.ID = Person.ID;
GO
I. 在遞迴 CTE 中使用分析函式
下列範例顯示在 CTE 遞迴部分中使用分析或彙總函式時可能會發生的錯誤。
DECLARE @t1 TABLE (itmID INT, itmIDComp INT);
INSERT @t1 VALUES (1, 10), (2, 10);
DECLARE @t2 TABLE (itmID INT, itmIDComp INT);
INSERT @t2 VALUES (3, 10), (4, 10);
WITH vw AS
(
SELECT itmIDComp, itmID FROM @t1
UNION ALL SELECT itmIDComp, itmID FROM @t2
),
r AS
(
SELECT t.itmID AS itmIDComp,
NULL AS itmID,
CAST (0 AS BIGINT) AS N,
1 AS Lvl
FROM (SELECT 1 UNION ALL SELECT 2 UNION ALL SELECT 3 UNION ALL SELECT 4) AS t(itmID)
UNION ALL
SELECT t.itmIDComp,
t.itmID,
ROW_NUMBER() OVER (PARTITION BY t.itmIDComp ORDER BY t.itmIDComp, t.itmID) AS N,
Lvl + 1
FROM r
INNER JOIN vw AS t
ON t.itmID = r.itmIDComp
)
SELECT Lvl, N FROM r;
下列結果是此查詢的預期結果。
Lvl N
1 0
1 0
1 0
1 0
2 4
2 3
2 2
2 1
下列結果是此查詢的實際結果。
Lvl N
1 0
1 0
1 0
1 0
2 1
2 1
2 1
2 1
N 會針對 CTE 遞迴部分的每個行程傳回 1,因為只有該遞迴層級的資料子集會傳遞給 ROWNUMBER。 對於此查詢遞迴部分的每個反覆運算,只有一個資料列會傳遞給 ROWNUMBER。
範例:Azure Synapse Analytics 和 Analytics Platform System (PDW)
J. 在 CTAS 陳述式內使用通用資料表運算式
下列範例會建立一個新的資料表,其中包含 Adventure Works Cycles 中每個銷售代表每年的銷售訂單總數。
USE AdventureWorks2022;
GO
CREATE TABLE SalesOrdersPerYear
WITH (DISTRIBUTION = HASH(SalesPersonID)) AS
-- Define the CTE expression name and column list.
WITH Sales_CTE (SalesPersonID, SalesOrderID, SalesYear) AS
-- Define the CTE query.
(
SELECT SalesPersonID, SalesOrderID, YEAR(OrderDate) AS SalesYear
FROM Sales.SalesOrderHeader
WHERE SalesPersonID IS NOT NULL
)
-- Define the outer query referencing the CTE name.
SELECT SalesPersonID, COUNT(SalesOrderID) AS TotalSales, SalesYear
FROM Sales_CTE
GROUP BY SalesYear, SalesPersonID
ORDER BY SalesPersonID, SalesYear;
GO
K. 在 CETAS 陳述式內使用通用資料表運算式
下列範例會建立一個新的外部資料表,其中包含 Adventure Works Cycles 中每個銷售代表每年的銷售訂單總數。
USE AdventureWorks2022;
GO
CREATE EXTERNAL TABLE SalesOrdersPerYear
WITH
(
LOCATION = 'hdfs://xxx.xxx.xxx.xxx:5000/files/Customer',
FORMAT_OPTIONS ( FIELD_TERMINATOR = '|' )
) AS
-- Define the CTE expression name and column list.
WITH Sales_CTE (SalesPersonID, SalesOrderID, SalesYear) AS
-- Define the CTE query.
(
SELECT SalesPersonID, SalesOrderID, YEAR(OrderDate) AS SalesYear
FROM Sales.SalesOrderHeader
WHERE SalesPersonID IS NOT NULL
)
-- Define the outer query referencing the CTE name.
SELECT SalesPersonID, COUNT(SalesOrderID) AS TotalSales, SalesYear
FROM Sales_CTE
GROUP BY SalesYear, SalesPersonID
ORDER BY SalesPersonID, SalesYear;
GO
L. 在陳述式中使用多個以逗號分隔的 CTE
下列範例示範如何在單一陳述式中包含兩個 CTE。 CTE 不可位於巢狀結構中 (無遞迴)。
WITH CountDate (TotalCount, TableName) AS
(
SELECT COUNT(datekey), 'DimDate' FROM DimDate
),
CountCustomer (TotalAvg, TableName) AS
(
SELECT COUNT(CustomerKey), 'DimCustomer' FROM DimCustomer
)
SELECT TableName, TotalCount
FROM CountDate
UNION ALL
SELECT TableName, TotalAvg FROM CountCustomer;