Nota
El acceso a esta página requiere autorización. Puede intentar iniciar sesión o cambiar directorios.
El acceso a esta página requiere autorización. Puede intentar cambiar los directorios.
Se aplica a:SQL Server
Azure SQL Database
Azure SQL Managed Instance
Azure Synapse Analytics
Analytics Platform System (PDW)
Punto de conexión de análisis SQL en Microsoft Fabric
Warehouse en Microsoft Fabric
Base de datos SQL de Microsoft Fabric
Especifica un conjunto de resultados temporal con nombre, conocido como expresión de tabla común (CTE). Esto se deriva de una consulta simple y se define dentro del ámbito de ejecución de una sola SELECT
instrucción , , INSERT
UPDATE
, , MERGE
o DELETE
. Esta cláusula también se puede usar en una CREATE VIEW
instrucción como parte de su instrucción de definición SELECT
. Una expresión de tabla común puede incluir referencias a ella misma. Esto se conoce como expresión de tabla común recursiva.
Convenciones de sintaxis de Transact-SQL
Sintaxis
[ WITH <common_table_expression> [ , ...n ] ]
<common_table_expression>::=
expression_name [ ( column_name [ , ...n ] ) ]
AS
( CTE_query_definition )
Argumentos
expression_name
Es un identificador válido de la expresión de tabla común.
expression_name debe ser diferente del nombre de cualquier otra expresión de tabla común definida en la misma cláusula WITH <common_table_expression>
, pero expression_name puede coincidir con el nombre de una vista o tabla base. Cualquier referencia a expression_name en la consulta usa la expresión de tabla común y no el objeto base.
column_name
Especifica un nombre de columna en la expresión de tabla común. No se permiten nombres duplicados en una misma definición de CTE. El número de nombres de columna especificado debe coincidir con el número de columnas del conjunto de resultados de CTE_query_definition. La lista de nombres de columna es opcional solamente si en la definición de la consulta se suministran nombres diferentes para todas las columnas resultantes.
CTE_query_definition
Especifica una SELECT
instrucción cuyo conjunto de resultados rellena la expresión de tabla común. La SELECT
instrucción para CTE_query_definition debe cumplir los mismos requisitos que para crear una vista, excepto que un CTE no puede definir otro CTE. Para obtener más información, vea la sección Comentarios y CREATE VIEW.
Si se define más de un CTE_query_definition , las definiciones de consulta deben estar unidas por uno de estos operadores de conjunto: UNION ALL
, UNION
, EXCEPT
o INTERSECT
.
Directrices de uso
Los resultados de la consulta de expresiones de tabla comunes no se materializan. Cada referencia externa al conjunto de resultados con nombre requiere que se vuelva a ejecutar la consulta definida. En el caso de las consultas que requieren varias referencias al conjunto de resultados con nombre, considere la posibilidad de usar un objeto temporal en su lugar.
No se puede ejecutar un procedimiento almacenado en una expresión de tabla común.
Para obtener instrucciones de uso sobre CTE recursivos y no recursivos, consulte las secciones siguientes.
- Directrices para expresiones de tabla comunes no recursivas
- Directrices para expresiones de tabla comunes recursivas
Directrices para expresiones de tabla comunes no recursivas
Nota:
Las instrucciones siguientes se aplican a expresiones de tabla comunes no recursivas. Para obtener instrucciones que se aplican a expresiones de tabla comunes recursivas, consulte Directrices para expresiones de tabla comunes recursivas.
Un CTE debe ir seguido de una sola SELECT
instrucción , , INSERT
UPDATE
, MERGE
, o DELETE
que haga referencia a algunas o todas las columnas de CTE. Una expresión CTE también se puede especificar en una instrucción CREATE VIEW
como parte de la instrucción SELECT
de definición de la vista.
Se pueden especificar varias definiciones de consulta de CTE en una CTE no recursiva. Las definiciones deben combinarse mediante uno de estos operadores de conjuntos: UNION ALL
, UNION
, INTERSECT
o EXCEPT
.
Una expresión CTE puede hacer referencia a ella misma y a otras expresiones CTE previamente definidas en la misma cláusula WITH
. No se permiten referencias hacia delante.
No se puede especificar más de una cláusula WITH
en una expresión CTE. Por ejemplo, si un CTE_query_definition contiene una subconsulta, esa subconsulta no puede contener una cláusula anidada WITH
que defina otro CTE.
Para obtener más información sobre los CTE anidados en Microsoft Fabric, vea Nested Common Table Expression (CTE) in Fabric data warehouseing (Transact-SQL).
En CTE_query_definition, no se pueden usar las siguientes cláusulas:
-
ORDER BY
(excepto cuando se especifica unaTOP
cláusula oOFFSET/FETCH
) INTO
-
OPTION
cláusula con sugerencias de consulta 1 FOR BROWSE
1 La OPTION
cláusula no se puede usar dentro de una definición de CTE. Solo se puede usar en la instrucción más SELECT
externa.
Cuando se utiliza una expresión CTE en una instrucción que forma parte de un lote, la instrucción que la precede debe ir seguida de punto y coma.
Una consulta que haga referencia a una CTE se puede utilizar para definir un cursor.
En la expresión CTE se puede hacer referencia a tablas de servidores remotos.
Al ejecutar un CTE, las sugerencias que hacen referencia a un CTE pueden entrar en conflicto con otras sugerencias que se detectan cuando el CTE accede a sus tablas subyacentes, de la misma manera que las sugerencias que hacen referencia a vistas en las consultas. En ese caso, la consulta devuelve un error.
Directrices para expresiones de tabla comunes recursivas
Nota:
Las instrucciones siguientes se aplican a la definición de una expresión de tabla común recursiva. Para obtener instrucciones que se aplican a CTE no recursivos, vea Directrices para expresiones de tabla comunes no recursivas.
La definición de la CTE recursiva debe contener al menos dos definiciones de consulta de CTE, un miembro no recursivo y un miembro recursivo. Se pueden definir varios miembros no recursivos y recursivos, aunque todas las definiciones de consultas de miembros no recursivos deben colocarse delante de la primera definición de miembro recursivo. Todas las definiciones de consulta de CTE son miembros no recursivos a menos que hagan referencia a la propia CTE.
Los miembros de delimitador deben combinarse mediante uno de estos operadores de conjunto: UNION ALL
, UNION
, INTERSECT
o EXCEPT
.
UNION ALL
es el único operador set permitido entre el último miembro delimitador y el primer miembro recursivo, y al combinar varios miembros recursivos.
El número de columnas de los miembros no recursivo y recursivo debe coincidir.
El tipo de datos de una columna del miembro recursivo debe ser igual al tipo de datos de la columna correspondiente en el miembro no recursivo.
La cláusula FROM de un miembro recursivo solo debe hacer referencia una vez a expression_name de CTE.
No se permiten los siguientes elementos en el parámetro CTE_query_definition de un miembro recursivo:
SELECT DISTINCT
GROUP BY
-
PIVOT
1 HAVING
- Agregación escalar
TOP
-
LEFT
,RIGHT
,OUTER JOIN
(INNER JOIN
se permite) - Subconsultas
- Una sugerencia aplicada a una referencia recursiva a una CTE dentro de CTE_query_definition.
1 Cuando el nivel de compatibilidad de la base de datos es 110 o superior. Consulte Cambios importantes en las características del motor de base de datos en SQL Server 2016.
Las instrucciones siguientes se aplican al uso de una expresión de tabla común recursiva:
Todas las columnas devueltas por la expresión CTE recursiva aceptan valores NULL independientemente de la nulabilidad de las columnas devueltas por las instrucciones
SELECT
participantes.Un CTE recursivo compuesto incorrectamente puede provocar un bucle infinito. Por ejemplo, si la definición de la consulta del miembro recursivo devuelve los mismos valores para las columnas primarias y secundarias, se crea un bucle infinito. Para evitar un bucle infinito, puede limitar el número de niveles de recursividad permitidos para una instrucción determinada mediante la
MAXRECURSION
sugerencia y un valor entre0
y32767
en laOPTION
cláusula de laINSERT
instrucción ,UPDATE
,DELETE
oSELECT
. De esta manera, se puede controlar la ejecución de la instrucción hasta que se resuelva el problema de código que genera el bucle. El valor predeterminado de todo el servidor es 100. Cuando se especifica 0, no se aplica ningún límite. Solo se puede especificar un valorMAXRECURSION
por instrucción. Para obtener más información, consulte Sugerencias de consulta.No se puede utilizar una vista que contenga una expresión de tabla común recursiva para actualizar datos.
Los cursores se pueden definir en consultas mediante CTE. La expresión CTE es el argumento de select_statement que define el conjunto de resultados del cursor. En el caso de las CTE recursivas únicamente se permiten los cursores de solo avance rápido y estáticos (de instantánea). Si se especifica otro tipo de cursor en una CTE recursiva, el tipo de cursor se convierte a estático.
En la expresión CTE se puede hacer referencia a tablas de servidores remotos. Si se hace referencia al servidor remoto en el miembro recursivo de la CTE, se crea una cola para cada tabla remota de manera que se pueda tener acceso local a las tablas repetidas veces. Si se trata de una consulta de CTE, las colas de índice/colas diferidas se muestran en el plan de consulta y tendrán el predicado adicional
WITH STACK
. Esta es una forma de confirmar la recursividad apropiada.Las funciones analíticas y de agregado de la parte recursiva del CTE se aplican al conjunto para el nivel de recursividad actual y no al conjunto para el CTE. Las funciones como
ROW_NUMBER
solo funcionan sobre el subconjunto de datos que les pasa el nivel de recursividad actual y no sobre todo el conjunto de datos pasados a la parte recursiva de la CTE. Para obtener más información, vea el ejemplo "I. Uso de funciones analíticas en una expresión CTE recursiva" a continuación.
Expresiones de tabla comunes en Azure Synapse Analytics and Analytics Platform System (PDW)
La implementación actual de CTE en Azure Synapse Analytics and Analytics Platform System (PDW) tiene las siguientes características y requisitos:
Una CTE se puede especificar en una instrucción
SELECT
.Una CTE se puede especificar en una instrucción
CREATE VIEW
.Una CTE se puede especificar en una instrucción
CREATE TABLE AS SELECT
(CTAS).Una CTE se puede especificar en una instrucción
CREATE REMOTE TABLE AS SELECT
(CRTAS).Una CTE se puede especificar en una instrucción
CREATE EXTERNAL TABLE AS SELECT
(CETAS).Se puede hacer referencia a una tabla remota desde una CTE.
Se puede hacer referencia a una tabla externa desde una CTE.
Se pueden especificar varias definiciones de consulta de CTE en una CTE.
Un CTE puede ir seguido de
SELECT
instrucciones ,INSERT
,UPDATE
,DELETE
oMERGE
.No se admiten expresiones de tabla comunes que incluyan referencias a sí mismas (expresiones de tabla comunes recursivas).
No se puede especificar más de una cláusula
WITH
en una expresión CTE. Por ejemplo, si una definición de consulta de CTE contiene una subconsulta, esta no puede contener ninguna cláusulaWITH
anidada que defina otra CTE.No
ORDER BY
se puede usar una cláusula en el CTE_query_definition, excepto cuando se especifica unaTOP
cláusula .Cuando se utiliza una expresión CTE en una instrucción que forma parte de un lote, la instrucción que la precede debe ir seguida de punto y coma.
Cuando se usa en instrucciones preparadas por
sp_prepare
, los CTE se comportan del mismo modo que otrasSELECT
instrucciones de APS PDW. Sin embargo, si los CTE se usan como parte de CETAS preparados porsp_prepare
, el comportamiento puede aplazarse de SQL Server y otras instrucciones PDW de APS debido a la forma en que se implementa el enlace parasp_prepare
. SiSELECT
hace referencia a CTE usa una columna incorrecta que no existe en CTE, pasasp_prepare
sin detectar el error, pero se produce el error ensp_execute
su lugar.
Ejemplos
Un. Creación de una expresión de tabla común
En el siguiente ejemplo se muestra el número total de pedidos de venta por año para cada representante de ventas en 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. Uso de una expresión de tabla común para limitar los recuentos y promedios en informes
En el siguiente ejemplo se muestra el número medio de pedidos de venta correspondiente a todos los años para los representantes de ventas.
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. Uso de varias definiciones de CTE en una sola consulta
En el ejemplo siguiente se muestra cómo definir más de una CTE en una sola consulta. Una coma se usa para separar las definiciones de consulta de CTE. La FORMAT
función , que se usa para mostrar los importes monetarios en un formato de moneda, se introdujo en 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;
A continuación se muestra un conjunto parcial de resultados.
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. Uso de una expresión de tabla común recursiva para mostrar varios niveles de recursividad
En el ejemplo siguiente se muestra la lista jerárquica de los directivos y de los empleados que tienen a su cargo. En el ejemplo se empieza creando y rellenando la tabla 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;
Uso de una expresión de tabla común recursiva para mostrar dos niveles de recursividad
En el ejemplo siguiente se muestran los directivos y los empleados que tienen a su cargo. El número de niveles devueltos está limitado a dos.
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;
Uso de una expresión de tabla común recursiva para mostrar una lista jerárquica
En el siguiente ejemplo se agregan los nombres del directivo y de los empleados, así como sus cargos respectivos. La jerarquía de directivos y empleados se resalta más mediante la aplicación de sangrías a cada nivel.
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;
Uso de MAXRECURSION para cancelar una instrucción
MAXRECURSION
se puede utilizar para impedir que una CTE recursiva con formato incorrecto entre en un bucle infinito. En el ejemplo siguiente se crea un bucle infinito intencionadamente y se utiliza la sugerencia MAXRECURSION
para limitar el número de niveles de recursividad a dos.
--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);
Una vez corregido el error de codificación, ya no se requiere MAXRECURSION
. En el siguiente ejemplo se muestra el código corregido.
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. Uso de una expresión de tabla común para recorrer selectivamente y paso a paso una relación recursiva en una instrucción SELECT
En el ejemplo siguiente se muestra la jerarquía de ensamblados y componentes de producto necesarios para fabricar la bicicleta para 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. Uso de una expresión CTE recursiva en una instrucción UPDATE
En el ejemplo siguiente se actualiza el PerAssemblyQty
valor de todas las partes que se usan para compilar el producto 'Road-550-W Yellow, 44' (ProductAssemblyID 800)
. La expresión de tabla común devuelve una lista jerárquica de los elementos que se utilizan para fabricar ProductAssemblyID 800
y los componentes que se utilizan para crear esos elementos, etc. Solo se modifican las filas devueltas por la expresión de tabla común.
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. Uso de varios miembros recursivos y no recursivos
En el ejemplo siguiente se utilizan varios miembros no recursivos y recursivos para devolver todos los antecesores de una persona especificada. Se crea una tabla y se insertan valores en ella para establecer la genealogía familiar devuelta por la CTE recursiva.
-- 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
Yo. Uso de funciones analíticas en una expresión CTE recursiva
En el siguiente ejemplo se muestra un error que puede producirse al utilizar una función analítica o de agregado en la parte recursiva de una 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;
Los siguientes resultados son los esperados para la consulta.
Lvl N
1 0
1 0
1 0
1 0
2 4
2 3
2 2
2 1
Los siguientes resultados son los resultados reales de la consulta.
Lvl N
1 0
1 0
1 0
1 0
2 1
2 1
2 1
2 1
N
devuelve 1 para cada paso de la parte recursiva del CTE, porque solo el subconjunto de datos para ese nivel de recursividad se pasa a ROWNUMBER
. Por cada iteración de la parte recursiva de la consulta solo se pasa una fila a ROWNUMBER
.
Ejemplos: Azure Synapse Analytics y Sistema de la plataforma de análisis (PDW)
J. Uso de una expresión de tabla común en una instrucción CTAS
En el siguiente ejemplo se crea una tabla que contiene el número total de pedidos de venta por año de cada representante de ventas en 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. Uso de una expresión de tabla común en una instrucción CETAS
En el siguiente ejemplo se crea una tabla externa que contiene el número total de pedidos de venta por año de cada representante de ventas en 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. Uso de varias expresiones CTE separadas por comas en una instrucción
En el siguiente ejemplo se muestra cómo incluir dos CTE en una misma instrucción. Las CTE no se pueden anidar (no debe haber recursividad).
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;