Table of Contents
PIVOT TABLE
Sometimes it is necessary to rotate results so that columns are presented horizontally and rows are presented vertically. This is known as creating a PivotTable, creating a cross-tab report, or rotating data.
Assume there is a table Pivot that has one row per quarter. A SELECT of Pivot reports the quarters vertically:
Year | Quarter | Amount |
---|---|---|
1990 | 1 | 1.1 |
1990 | 2 | 1.2 |
1990 | 3 | 1.3 |
1990 | 4 | 1.4 |
1991 | 1 | 2.1 |
1991 | 2 | 2.2 |
1991 | 3 | 2.3 |
1991 | 4 | 2.4 |
These are the statements used to create the Pivot table and populate it with the data from the first table:
CREATE TABLE Pivot ( YEAR INT, Quarter INT, Amount FLOAT ); INSERT INTO Pivot VALUES (1990, 1, 1.1); INSERT INTO Pivot VALUES (1990, 2, 1.2); INSERT INTO Pivot VALUES (1990, 3, 1.3); INSERT INTO Pivot VALUES (1990, 4, 1.4); INSERT INTO Pivot VALUES (1991, 1, 2.1); INSERT INTO Pivot VALUES (1991, 2, 2.2); INSERT INTO Pivot VALUES (1991, 3, 2.3); INSERT INTO Pivot VALUES (1991, 4, 2.4);
This is the SELECT statement used to create the rotated results:
SELECT YEAR, SUM(CASE Quarter WHEN 1 THEN Amount ELSE 0 END) AS Q1, SUM(CASE Quarter WHEN 2 THEN Amount ELSE 0 END) AS Q2, SUM(CASE Quarter WHEN 3 THEN Amount ELSE 0 END) AS Q3, SUM(CASE Quarter WHEN 4 THEN Amount ELSE 0 END) AS Q4 FROM Pivot GROUP BY YEAR
This SELECT statement also handles a table in which there are multiple rows for each quarter. The GROUP BY combines all rows in Pivot for a given year into a single row in the output. When the grouping operation is being performed, the CASE functions in the SUM aggregates are applied in such a way that the Amount values for each quarter are added into the proper column in the result set and 0 is added to the result set columns for the other quarters.
If the results of this SELECT statement are used as input to a spreadsheet, it is easy for the spreadsheet to calculate a total for each year. When the SELECT is used from an application it may be easier to enhance the SELECT statement to calculate the yearly total. For example:
SELECT P1.*, (P1.Q1 + P1.Q2 + P1.Q3 + P1.Q4) AS YearTotal FROM (SELECT YEAR, SUM(CASE P.Quarter WHEN 1 THEN P.Amount ELSE 0 END) AS Q1, SUM(CASE P.Quarter WHEN 2 THEN P.Amount ELSE 0 END) AS Q2, SUM(CASE P.Quarter WHEN 3 THEN P.Amount ELSE 0 END) AS Q3, SUM(CASE P.Quarter WHEN 4 THEN P.Amount ELSE 0 END) AS Q4 FROM Pivot AS P GROUP BY P.Year) AS P1
Pivot Transformation
The Pivot transformation makes a normalized data set into a less normalized but more compact version by pivoting the input data on a column value. For example, a normalized Orders data set that lists customer name, product, and quantity purchased typically has multiple rows for any customer who purchased multiple products, with each row for that customer showing order details for a different product. By pivoting the data set on the product column, the Pivot transformation can output a data set with a single row per customer. That single row lists all the purchases by the customer, with the product names shown as column names, and the quantity shown as a value in the product column. Because not every customer purchases every product, many columns may contain null values.
When a dataset is pivoted, input columns perform different roles in the pivoting process. A column can participate in the following ways:
- The column is passed through unchanged to the output. Because many input rows can result only in one output row, the transformation copies only the first input value for the column.
- The column acts as the key or part of the key that identifies a set of records.
- The column defines the pivot. The values in this column are associated with columns in the pivoted data set.
- The column contains values that are placed in the columns that the pivot creates.
The following diagram shows a data set before the data is pivoted on the Product column.
Usage | SetKey | PivotKey | PivotedVal |
---|---|---|---|
ColName | Cust | Product | Qty |
Data Records | Kate | Ham | 2 |
Kate | Soda | 6 | |
Kate | Milk | 1 | |
Kate | Beer | 12 | |
Fred | Milk | 3 | |
Fred | Beer | 24 | |
Fred | Chips | 2 |
The following diagram shows a data set after the data has been pivoted on the Product column.
Source Column | Cust | Qty | Qty | Qty | Qty | Qty |
---|---|---|---|---|---|---|
Pivot Key Value | Ham | Soda | Milk | Beer | Chips | |
Column Name | Cust | Ham | Soda | Milk | Beer | Chips |
Data Records | Kate | 2 | 6 | 1 | 12 | |
Fred | 3 | 24 | 2 |
To pivot data efficiently, which means creating as few records in the output dataset as possible, the input data must be sorted on the pivot column. If the data is not sorted, the Pivot transformation might generate multiple records for each value in the set key, which is the column that defines set membership. For example, if the dataset is pivoted on a Name column but the names are not sorted, the output dataset could have more than one row for each customer, because a pivot occurs every time that the value in Name changes.
The Pivot transformation uses the properties on its input and output columns to define the pivot operation.
The Pivot transformation has one input, one regular output, and one error output.
SQL syntax
FROM table_reference [, table_reference ]* table_reference : non_join_table | joined_table | pivoted_table pivoted_table : table_reference PIVOT pivot_clause AS table_alias pivot_clause : ( aggregate_function ( value_column ) FOR column_name IN ( column_list ) )
Using PIVOT more examples
CREATE TABLE tblCar( Car VARCHAR (10), Color VARCHAR (10), Cnt BYTE);
INSERT INTO tblCar VALUES ('Toyota','Red',1), ('Toyota','Green',2), ('Nissan','Blue',3), ('Nissan','Red',1), ('Honda','Green',5), ('Toyota','Red',2), ('Toyota','Blue',3), ('Honda','Red',1), ('Nissan','Red',2);
SELECT Car, RED, GREEN, BLUE FROM (SELECT * FROM tblCar) AS tblSRC PIVOT (SUM(Cnt) FOR Color IN (RED, GREEN, BLUE)) AS tblPVT;
Car | Red | Green | Blue |
---|---|---|---|
Honda | 1 | 5 | 0 |
Nissan | 3 | 0 | 3 |
Toyota | 3 | 2 | 3 |