FROM: uso de PIVOT y UNPIVOT
Se aplica a: SQL Server Azure SQL Database Azure SQL Instancia administrada Azure Synapse Analytics Analytics Platform System (PDW)
Se pueden usar los operadores relacionales PIVOT
y UNPIVOT
para modificar una expresión con valores de tabla en otra tabla. PIVOT
gira una expresión con valores de tabla al convertir los valores únicos de una columna en la expresión en varias columnas en la salida. Y PIVOT
ejecuta agregaciones donde se requieren en los valores de columna restantes que se desean en la salida final. UNPIVOT
realiza la operación contraria a PIVOT girando las columnas de una expresión con valores de tabla a valores de columna.
La sintaxis de PIVOT
es más sencilla y legible que la sintaxis que se puede especificar en una serie compleja de instrucciones SELECT...CASE
. Para obtener una descripción completa de la sintaxis de PIVOT
, vea FROM (Transact-SQL).
Sintaxis
La sintaxis siguiente resume cómo se usa el operador PIVOT
.
SELECT <non-pivoted column>,
[first pivoted column] AS <column name>,
[second pivoted column] AS <column name>,
...
[last pivoted column] AS <column name>
FROM
(<SELECT query that produces the data>)
AS <alias for the source query>
PIVOT
(
<aggregation function>(<column being aggregated>)
FOR
[<column that contains the values that will become column headers>]
IN ( [first pivoted column], [second pivoted column],
... [last pivoted column])
) AS <alias for the pivot table>
<optional ORDER BY clause>;
Observaciones
Los identificadores de columna de la cláusula UNPIVOT
siguen la intercalación del catálogo. Para SQL Database, la intercalación es siempre SQL_Latin1_General_CP1_CI_AS
. Para las bases de datos parcialmente independientes de SQL Server, la intercalación es siempre Latin1_General_100_CI_AS_KS_WS_SC
. Si la columna se combina con otras columnas, se necesita una cláusula COLLATE (COLLATE DATABASE_DEFAULT
) para evitar conflictos.
En los grupos de Microsoft Fabric y Azure Synapse Analytics, se produce un error en las consultas con el operador PIVOT si hay un GROUP BY en la salida de columna no dinámica de PIVOT. Como solución alternativa, quite la columna no dinámica de GROUP BY. Los resultados de la consulta son los mismos, ya que esta cláusula GROUP BY está duplicada.
Ejemplo PIVOT básico
En el ejemplo de código siguiente se genera una tabla de dos columnas con cuatro filas.
USE AdventureWorks2022;
GO
SELECT DaysToManufacture, AVG(StandardCost) AS AverageCost
FROM Production.Product
GROUP BY DaysToManufacture;
El conjunto de resultados es el siguiente:
DaysToManufacture AverageCost
----------------- -----------
0 5.0885
1 223.88
2 359.1082
4 949.4105
No hay productos definidos con tres DaysToManufacture
.
En el código siguiente se muestra el mismo resultado, dinamizado para que los valores de DaysToManufacture
se conviertan en encabezados de columna. Se proporciona una columna para tres [3]
días, aunque los resultados son NULL
.
-- Pivot table with one row and five columns
SELECT 'AverageCost' AS Cost_Sorted_By_Production_Days,
[0], [1], [2], [3], [4]
FROM
(
SELECT DaysToManufacture, StandardCost
FROM Production.Product
) AS SourceTable
PIVOT
(
AVG(StandardCost)
FOR DaysToManufacture IN ([0], [1], [2], [3], [4])
) AS PivotTable;
El conjunto de resultados es el siguiente:
Cost_Sorted_By_Production_Days 0 1 2 3 4
------------------------------ ----------- ----------- ----------- ----------- -----------
AverageCost 5.0885 223.88 359.1082 NULL 949.4105
Ejemplo PIVOT complejo
Un escenario habitual en el que PIVOT
puede ser útil es cuando se desea generar informes de tabulación cruzada para proporcionar un resumen de los datos. Por ejemplo, suponga que desea consultar la tabla PurchaseOrderHeader
en la base de datos de ejemplo AdventureWorks2022
para determinar el número de pedidos de compra colocados por ciertos empleados. En la siguiente consulta se proporciona este informe, ordenado por proveedor.
USE AdventureWorks2022;
GO
SELECT VendorID, [250] AS Emp1, [251] AS Emp2, [256] AS Emp3, [257] AS Emp4, [260] AS Emp5
FROM
(SELECT PurchaseOrderID, EmployeeID, VendorID
FROM Purchasing.PurchaseOrderHeader) p
PIVOT
(
COUNT (PurchaseOrderID)
FOR EmployeeID IN
( [250], [251], [256], [257], [260] )
) AS pvt
ORDER BY pvt.VendorID;
A continuación se muestra un conjunto parcial de resultados.
VendorID Emp1 Emp2 Emp3 Emp4 Emp5
----------- ----------- ----------- ----------- ----------- -----------
1492 2 5 4 4 4
1494 2 5 4 5 4
1496 2 4 4 5 5
1498 2 5 4 4 4
1500 3 4 4 5 4
Los resultados devueltos por esta instrucción de subselección se dinamizan en la columna EmployeeID
.
SELECT PurchaseOrderID, EmployeeID, VendorID
FROM PurchaseOrderHeader;
Los valores únicos devueltos por la columna EmployeeID
se convierten en campos en el conjunto de resultados finales. Como tal, hay una columna para cada número de EmployeeID
especificado en la cláusula dinámica: en este caso los empleados 250
, 251
, 256
, 257
y 260
. La columna PurchaseOrderID
se utiliza como columna de valores, respecto a la que se ordenan las columnas del resultado final, denominadas columnas de agrupamiento. En este caso, las columnas de agrupamiento se agregan mediante la función COUNT
. Tenga presente que aparece un mensaje de advertencia que indica que los valores NULL que aparecen en la columna PurchaseOrderID
no se tuvieron en cuenta cuando se contabilizó COUNT
para cada empleado.
Importante
Cuando se usan funciones de agregado con PIVOT
, la presencia de valores NULL en la columna de valores no se tiene en cuenta cuando se calcula una agregación.
Ejemplo de UNPIVOT
UNPIVOT
realiza casi la operación inversa de PIVOT
, girando columnas en filas. Suponga que la tabla producida en el ejemplo anterior se almacena en la base de datos como pvt
y que desea girar los identificadores de columna Emp1
, Emp2
, Emp3
, Emp4
y Emp5
a valores de fila que correspondan a un determinado proveedor. Como tal, debe identificar dos columnas adicionales. La columna que contendrá los valores de columna que se están girando (Emp1
, Emp2
,...) se denominará Employee
y la columna que contendrá los valores que existen actualmente en las columnas que se giran se denominará Orders
. Estas columnas corresponden a pivot_column y value_column, respectivamente, en la definición de Transact-SQL. Aquí está la consulta.
-- Create the table and insert values as portrayed in the previous example.
CREATE TABLE pvt (VendorID INT, Emp1 INT, Emp2 INT,
Emp3 INT, Emp4 INT, Emp5 INT);
GO
INSERT INTO pvt VALUES (1,4,3,5,4,4);
INSERT INTO pvt VALUES (2,4,1,5,5,5);
INSERT INTO pvt VALUES (3,4,3,5,4,4);
INSERT INTO pvt VALUES (4,4,2,5,5,4);
INSERT INTO pvt VALUES (5,5,1,5,5,5);
GO
-- Unpivot the table.
SELECT VendorID, Employee, Orders
FROM
(SELECT VendorID, Emp1, Emp2, Emp3, Emp4, Emp5
FROM pvt) p
UNPIVOT
(Orders FOR Employee IN
(Emp1, Emp2, Emp3, Emp4, Emp5)
)AS unpvt;
GO
A continuación se muestra un conjunto parcial de resultados.
VendorID Employee Orders
----------- ----------- ------
1 Emp1 4
1 Emp2 3
1 Emp3 5
1 Emp4 4
1 Emp5 4
2 Emp1 4
2 Emp2 1
2 Emp3 5
2 Emp4 5
2 Emp5 5
...
Tenga en cuenta que UNPIVOT
no es exactamente lo contrario a PIVOT
. PIVOT
realiza una agregación y combina posibles múltiples filas en una sola fila en la salida. UNPIVOT
no reproduce el resultado de la expresión con valores de tabla original porque las filas se han combinado. Además, los valores null de la entrada de UNPIVOT
desaparecen en la salida. Cuando los valores desaparecen, se muestra que puede haber habido valores null originales en la entrada antes de la operación PIVOT
.
En la vista Sales.vSalesPersonSalesByFiscalYears
de la base de datos de ejemplo AdventureWorks2022
se usa PIVOT
para devolver el total de ventas de cada vendedor, para cada año fiscal. Para generar el script de la vista en SQL Server Management Studio, en el Explorador de objetos, localice la vista en la carpeta Views de la base de datos AdventureWorks2022
. Haga clic con el botón derecho en el nombre de la vista y después seleccione Incluir vista como.
Pasos siguientes
Comentarios
https://aka.ms/ContentUserFeedback.
Próximamente: A lo largo de 2024 iremos eliminando gradualmente GitHub Issues como mecanismo de comentarios sobre el contenido y lo sustituiremos por un nuevo sistema de comentarios. Para más información, vea:Enviar y ver comentarios de