In Oracle databases, row-to-column conversion usually involves converting multiple rows of data in a table into columns, which is very common in data reporting and analysis. Oracle databases do not have direct functions or commands that can directly complete this transformation, but it can be implemented in several different ways, including usingCASE
Statement,PIVOT
Operations and dynamic SQL. Here are some ways to implement row-to-column conversion:
1. Row and Row
1. UseCASE
Statement
We first check the total salary data for each position in each department
SELECT job,deptno,SUM(sal) AS sum_sal
FROM scott.emp
WHERE deptno IS NOT NULL
GROUP BY job,deptno
ORDER BY job, deptno;
JOB DEPTNO SUM_SAL
--------- ------ ----------
ANALYST 20 6000
CLERK 10 1300
CLERK 20 1900
CLERK 30 950
MANAGER 10 2450
MANAGER 20 2975
MANAGER 30 2850
PRESIDENT 10 5000
SALESMAN 30 5600
CASE
The statement can be inSELECT
Convert row data into columns according to the conditions in the query. This approach is very effective when dealing with a fixed number of columns.
SELECT job,
SUM(CASE deptno WHEN 10 THEN sal ELSE 0 END) deptno_10_sal,
SUM(CASE deptno WHEN 20 THEN sal ELSE 0 END) deptno_20_sal,
SUM(CASE deptno WHEN 30 THEN sal ELSE 0 END) deptno_30_sal
FROM scott.emp
GROUP BY job;
-- Execute result set
JOB DEPTNO_10_SAL DEPTNO_20_SAL DEPTNO_30_SAL
--------- ------------- ------------- -------------
CLERK 1300 1900 950
SALESMAN 0 0 5600
PRESIDENT 5000 0 0
MANAGER 2450 2975 2850
ANALYST 0 6000 0
2. UsePIVOT
operate
Oracle 11g and later versions have been introducedPIVOT
Operator, specifically used to convert row data into column data. It makes the conversion process more concise and intuitive.
The syntax of the PIVOT function is
SELECT ...
FROM ...
PIVOT
( pivot_clause
pivot_for_clause
pivot_in_clause )
WHERE ...
in:
- pivot_clause means aggregation operation on a certain column
- pivot_for_clause means grouped and columns that need to be converted into rows
- pivot_in_clause represents the value range of column name for row-to-column
The PIVOT statement is between the FROM clause and the WHERE clause. We can filter the result set after row-to-column conversion through the WHERE clause, as follows:
WITH cte_data AS (
SELECT job, deptno, sal
FROM scott.emp
WHERE deptno IS NOT NULL
)
SELECT *
FROM cte_data
PIVOT (
SUM(sal) -- pivot_clause
FOR deptno -- pivot_for_clause
IN (10,20,30) -- pivot_in_clause
);
-- Execute result set
JOB 10 20 30
--------- ---------- ---------- ----------
CLERK 1300 1900 950
SALESMAN 5600
PRESIDENT 5000
MANAGER 2450 2975 2850
ANALYST 6000
From the above output, we can see that the column names of rows and columns are all numbers. We can specify an alias for these column names and specify where conditions are as follows:
WITH cte_data AS (
SELECT job, deptno, sal
FROM scott.emp
WHERE deptno IS NOT NULL
)
SELECT *
FROM cte_data
PIVOT (
SUM(sal) -- Sum and aggregate sal
FOR deptno -- Deptno needs to be row-to-row-redirected
IN (10 AS DEPTNO_10_sal,
20 AS DEPTNO_20_sal,
30 AS DEPTNO_30_sal) -- The range of column names after row-to-column conversion
)
WHERE job IN ('ANALYST','CLERK','SALESMAN');
-- Execute result set
JOB DEPTNO_10_SAL DEPTNO_20_SAL DEPTNO_30_SAL
--------- ------------- ------------- -------------
CLERK 1300 1900 950
SALESMAN 5600
ANALYST 6000
Notice:PIVOT
In-houseIN
The clause lists all possible things you want to convert to a columndeptno
value.
During aggregation, we can add multiple aggregate functions, as follows
WITH cte_data AS (
SELECT job, deptno, sal
FROM scott.emp
WHERE deptno IS NOT NULL
)
SELECT *
FROM cte_data
PIVOT (
SUM(sal) AS sum_sal, -- Sum and aggregate sal
COUNT(sal) AS cnt -- Aggregate the total number of sals
FOR deptno -- Deptno needs to be row-to-row-redirected
IN (10 AS DEPTNO_10_sal,
20 AS DEPTNO_20_sal,
30 AS DEPTNO_30_sal) -- The range of column names after row-to-column conversion
)
WHERE job IN ('ANALYST','CLERK','SALESMAN');
-- Execute result set
JOB DEPTNO_10_SAL_SUM_SAL DEPTNO_10_SAL_CNT DEPTNO_20_SAL_SUM_SAL DEPTNO_20_SAL_CNT DEPTNO_30_SAL_SUM_SAL DEPTNO_30_SAL_CNT
--------- --------------------- ----------------- --------------------- ----------------- --------------------- -----------------
CLERK 1300 1 1900 2 950 1
SALESMAN 0 0 5600 4
ANALYST 0 6000 2 0
pivot_for_clause can also use multiple fields, as follows:
WITH cte_data AS (
SELECT job, deptno, sal
FROM scott.emp
WHERE deptno IS NOT NULL
)
SELECT *
FROM cte_data
PIVOT (
SUM(sal) AS sum, -- Sum and aggregate sal
COUNT(sal) AS cnt -- Aggregate the total number of sals
FOR (deptno, job) -- Deptno and job need to be row-to-row-redirected
IN ((30, 'ANALYST') AS deptno_30_analyst,
(30, 'CLERK') AS deptno_30_clerk,
(30, 'SALESMAN') AS deptno_30_salesman) -- The range of column names after row-to-column conversion
);
-- Execute result set
DEPTNO_30_ANALYST_SUM DEPTNO_30_ANALYST_CNT DEPTNO_30_CLERK_SUM DEPTNO_30_CLERK_CNT DEPTNO_30_SALESMAN_SUM DEPTNO_30_SALESMAN_CNT
--------------------- --------------------- ------------------- ------------------- ---------------------- ----------------------
0 950 1 5600 4
3. Dynamic SQL
Dynamic SQL can be used to build and execute when the number or name of the column is unknown at the time of query execution (e.g., based on dynamic changes in data in the table).PIVOT
Query orCASE
Statement. This usually involves using PL/SQL to write a program that can dynamically build and execute SQL statements.
in conclusion
Which method to choose depends on your specific needs, such as whether the number of columns is fixed, whether the columns need to be dynamically determined at runtime, etc. For simple fixed column conversion,CASE
statement orPIVOT
Operation may be enough. For more complex dynamic column transformations, you may need to use dynamic SQL.
2. Column and turn rows
The column conversion operation in Oracle database, that is, converting column data in a table into row data, is a common data processing requirement. Here are some ways to implement Oracle column conversion:
Prepare test tables and data
CREATE table scott.t_pivoted_emp
AS
SELECT * FROM (
WITH cte_data AS (
SELECT job, deptno, sal
FROM scott.emp
WHERE deptno IS NOT NULL
)
SELECT *
FROM cte_data
PIVOT (
SUM(sal) sum
FOR deptno
IN (10 AS deptno_10_sal,
20 AS deptno_20_sal,
30 AS deptno_30_sal)
)
);
SQL> select * from scott.t_pivoted_emp;
JOB DEPTNO_10_SAL_SUM DEPTNO_20_SAL_SUM DEPTNO_30_SAL_SUM
--------- ----------------- ----------------- -----------------
CLERK 1300 1900 950
SALESMAN 5600
PRESIDENT 5000
MANAGER 2450 2975 2850
ANALYST 6000
1. Use CASE statements and UNION ALL
When the UNPIVOT function does not meet the requirements, you can use the CASE statement combined with UNION ALL to achieve more complex column conversion. Although this method is more cumbersome, it is more flexible.
SELECT * FROM (
SELECT job, 10 as deptno,DEPTNO_10_SAL_SUM as salary FROM scott.t_pivoted_emp
UNION
SELECT job, 20 as deptno,DEPTNO_20_sal_sum as salary FROM scott.t_pivoted_emp
UNION
SELECT job, 30 as deptno,DEPTNO_30_sal_sum as salary FROM scott.t_pivoted_emp
)
WHERE salary IS NOT NULL
ORDER BY job, deptno;
JOB DEPTNO SALARY
--------- ---------- ----------
ANALYST 20 6000
CLERK 10 1300
CLERK 20 1900
CLERK 30 950
MANAGER 10 2450
MANAGER 20 2975
MANAGER 30 2850
PRESIDENT 10 5000
SALESMAN 30 5600
9 rows selected
This method generates a row collection for each column, which contains the name and value of the original column.
2. Use UNPIVOT function
UNPIVOT is a function provided by Oracle that is specifically used for column conversion. It can convert columns in a table into rows and specify which columns need to be converted. The basic syntax is as follows:
SELECT job,deptno,salary
FROM scott.t_pivoted_emp
UNPIVOT
(salary
FOR deptno
IN (DEPTNO_10_SAL_SUM AS 10,DEPTNO_20_SAL_SUM AS 20,DEPTNO_30_SAL_SUM AS 30))
ORDER BY job, deptno;
JOB DEPTNO SALARY
--------- ---------- ----------
ANALYST 20 6000
CLERK 10 1300
CLERK 20 1900
CLERK 30 950
MANAGER 10 2450
MANAGER 20 2975
MANAGER 30 2850
PRESIDENT 10 5000
SALESMAN 30 5600
In this example,DEPTNO_10_SAL_SUM
,DEPTNO_20_SAL_SUM
and DEPTNO_30_SAL_SUM
will be converted into rows, andJOB
and converted values (byvalue
Columns represent) will remain as columns.
3. Custom functions and procedures
For very specific needs, you may need to write custom PL/SQL functions or procedures to implement column-to-line conversion. This approach provides maximum flexibility, but also requires corresponding programming skills and a deep understanding of Oracle PL/SQL.
Summarize
In practical applications, the most suitable column-to-line conversion method should be selected according to specific needs and data structures. For most common scenarios, UNPIVOT function and CASE statement combined with UNION ALL are two simple and effective solutions. For more complex scenarios, you may need to consider usingCONNECT BY LEVEL
Or write custom functions.