SQL Server 如何选择列作为行?

xkftehaa  于 2023-01-20  发布在  其他
关注(0)|答案(6)|浏览(146)

所以,我一直在四处寻找,我发现了一些与我的问题相似的东西,但我需要更多的帮助来获得真正的解决方案。
我试图构造一个查询,它将返回2列数据,第一列应该是列名本身的列表,第二列应该是该列的值。
从视觉上看它应该是这样的

Column1      Column2
-------      -------
columnA      value_of_columnA
columnB      value_of_columnB
...          ...

我非常肯定这需要动态SQL来实现,但是我甚至不知道如何开始创建查询。
任何帮助都是感激的!

ztmd8pv5

ztmd8pv51#

这应该适用于任何表,但在我的示例中,我只创建了一个测试表。您需要在@YourTableName中设置表名。此外,您需要设置@YourTableWhere以将结果限制为一行,否则输出看起来会很奇怪,因为多行混合在一起。
试试这个:

BEGIN TRY
CREATE TABLE YourTestTable
(RowID       int primary key not null identity(1,1)
,col1        int null
,col2        varchar(30)
,col3        varchar(20)
,col4        money
,StatusValue char(1)
,xyz_123     int
)
INSERT INTO YourTestTable (col1,col2,col3,col4,StatusValue,xyz_123) VALUES (1234,'wow wee!','this is a long test!',1234.56,'A',98765)
INSERT INTO YourTestTable (col1,col2,col3,col4,StatusValue,xyz_123) VALUES (543,'oh no!','short test',0,'I',12)

END TRY BEGIN CATCH END CATCH

select * from YourTestTable

DECLARE @YourTableName   varchar(1000)
DECLARE @YourTableWhere  varchar(1000)
DECLARE @YourQuery       varchar(max)

SET @YourTableName='YourTestTable'
set @YourTableWhere='y.RowID=1'

SELECT
    @YourQuery = STUFF(
                       (SELECT
                            ' UNION '
                            + 'SELECT '''+COLUMN_NAME+''', CONVERT(varchar(max),'+COLUMN_NAME+') FROM '+@YourTableName+' y'+ISNULL('  WHERE '+@YourTableWhere,'')
                            FROM INFORMATION_SCHEMA.COLUMNS
                            WHERE table_name = @YourTableName
                            FOR XML PATH('')
                       ), 1, 7, ''
                      )

PRINT @YourQuery  

EXEC (@YourQuery)

输出:

RowID       col1        col2                           col3                 col4                  StatusValue xyz_123
----------- ----------- ------------------------------ -------------------- --------------------- ----------- -----------
1           1234        wow wee!                       this is a long test! 1234.56               A           98765
2           543         oh no!                         short test           0.00                  I           12

SELECT 'RowID', CONVERT(varchar(max),RowID) FROM YourTestTable y  WHERE y.RowID=1 UNION SELECT 'col1', CONVERT(varchar(max),col1) FROM YourTestTable y  WHERE y.RowID=1 UNION SELECT 'col2', CONVERT(varchar(max),col2) FROM YourTestTable y  WHERE y.RowID=1 UNION SELECT 'col3', CONVERT(varchar(max),col3) FROM YourTestTable y  WHERE y.RowID=1 UNION SELECT 'col4', CONVERT(varchar(max),col4) FROM YourTestTable y  WHERE y.RowID=1 UNION SELECT 'StatusValue', CONVERT(varchar(max),StatusValue) FROM YourTestTable y  WHERE y.RowID=1 UNION SELECT 'xyz_123', CONVERT(varchar(max),xyz_123) FROM YourTestTable y  WHERE y.RowID=1

----------- ------------------------
col1        1234
col2        wow wee!
col3        this is a long test!
col4        1234.56
RowID       1
StatusValue A
xyz_123     98765
    • 编辑**

为了与SQL Server 2000兼容,您应该能够将varchar(max)替换为varchar(8000),并使用此语句替换上面代码中的SELECT @YourQuery查询:

SELECT
    @YourQuery=ISNULL(@YourQuery+' UNION ','')
        + 'SELECT '''+COLUMN_NAME+''', CONVERT(varchar(max),'+COLUMN_NAME+') FROM '+@YourTableName+' y'+ISNULL('  WHERE '+@YourTableWhere,'')
    FROM INFORMATION_SCHEMA.COLUMNS
    WHERE table_name = @YourTableName
0yg35tkg

0yg35tkg2#

select column_name,* from information_schema.columns
 where table_name = 'TheTableName'
order by ordinal_position
lc8prwob

lc8prwob3#

你总是可以做这样的事

SELECT 'Column_Name' AS ColumnName, 
  (SELECT TOP 1 Column_Name FROM Table tbl2 WHERE tbl.ID = tbl2.ID)
FROM Table tbl
tvokkenx

tvokkenx4#

我对这个问题的回答在SQL Server 2000中会更容易,因为它不使用SQL Server 2005的XML功能。

mzmfm0qo

mzmfm0qo5#

你不太清楚你是如何展示你的报告以及你是用什么来生成它的。你是用查询工具的直接结果来生成你的“报告”吗?在这种情况下,我认为你是在用螺丝刀磅钉子。使用正确的工具来完成这项工作。
SQL语言不应该直接用于设置您的表示数据以生成报表。实际上,这是一个愚蠢的想法。事实上,您可以使用直接的SQL语句编写报表并不意味着您应该这样做。
您确实应该使用自己编写的应用程序或Crystal Reports之类的报表生成工具来生成报表。

  • 自己编写的应用程序:* 如果使用游标对象查询数据库,只需从游标对象中获取列名即可。

报告生成工具:通常,它们提供表示可能出现的动态数据的工具。
不管怎样,我觉得你需要重新考虑你的方法。

b1zrtrql

b1zrtrql6#

这应该可以在SQL 2008中使用
选择列名列值您的表取消透视(列值用于列名在(列A列B列C列D))作为取消透视表

相关问题