在列查询中使用列值作为列名

时间:2017-10-10 18:36:11

标签: sql sql-server sql-server-2008 tsql unpivot

我正在使用一个旧数据库,该数据库有一个包含其他表中字段名称的表。

所以我有这个结构:

Field_ID | Field_Name
*********************
1 | Col1  
2 | Col2  
3 | Col3  
4 | Col4  

我需要提取此字段元数据的列表以及给定用户的该字段的值。所以我需要:

Field_ID | Field_Name | Value  
1 | Col1 | ValueOfCol1onADiffTable  
2 | Col2 | ValueOfCol2onADiffTable  
3 | Col3 | ValueOfCol3onADiffTable  
4 | Col4 | ValueOfCol4onADiffTable  

我想在子查询中使用Field_Name来提取该值,但无法弄清楚如何让SQL将Field_Name评估为子查询中的列。

这样的事情:

select
        Field_ID
      ,Field_Name
      ,(SELECT f.Field_Name from tblUsers u 
    where u.User_ID = @userId) as value
from 
  dbo.tblFields f 

但是这只会在值列中返回Field_Name,而不是它的值。

我是否需要将子查询放在单独的函数中并进行评估?还是某种动态SQL?

1 个答案:

答案 0 :(得分:1)

在SQL Server中,这需要动态SQL和UNPIVOT表示法。 see working demo

create table tblFields (Field_ID int ,Field_Name varchar(10));
insert into tblFields values
(1,'Col1')
,(2,'Col2')
,(3,'Col3')
,(4,'Col4');
declare @userId int
set @userId=1

create table tblUsers (User_ID int, col1 varchar(10),col2 varchar(10));
insert into tblUsers values
(1, 10,100),
(2,20,200);

declare @collist varchar(max)
declare @sqlquery varchar(max)

select @collist= COALESCE(@collist + ', ', '') + Field_Name
from dbo.tblFields
where exists (
       select * from sys.columns c join sys.tables t 
       on c.object_id=t.object_id and t.name='tblUsers'
       and c.name =Field_Name)
select @sqlquery=
' select Field_ID ,Field_Name, value '+
' from dbo.tblFields f  Join '+
' ( select  * from '+
'( select * '+
' from tblUsers u '+
' where u.User_ID = '+ cast(@userId as varchar(max)) +
' ) src '+
'unpivot ( Value for field in ('+ @collist+')) up )t'+
' on t.field =Field_Name'
exec(@sqlquery)