Tsql 动态字段定义-这可以在T-SQL中完成吗?

Tsql 动态字段定义-这可以在T-SQL中完成吗?,tsql,views,Tsql,Views,我有一个我正在努力实现的要求。如果可能的话,我想用原生T-SQL实现这一点 我有以下表格: CUSTOMER ======== ID, Name FIELDDEF ======== ID, Name FieldType (Char T, N, D for Text, Number or Date) CUSTOMERFIELD ============= ID, CustomerID, FieldDefID, CaptureDate, ValueText, ValueNumber,

我有一个我正在努力实现的要求。如果可能的话,我想用原生T-SQL实现这一点

我有以下表格:

CUSTOMER
========
ID,
Name

FIELDDEF
========
ID,
Name
FieldType   (Char  T, N, D   for Text, Number or Date)

CUSTOMERFIELD
=============
ID,
CustomerID,
FieldDefID,
CaptureDate,
ValueText,
ValueNumber,
ValueDate
基本上,这些表的目的是提供一个可扩展的自定义字段系统。其思想是用户创建新的字段定义,可以是文本、数字或日期字段。然后,它们在ValueText、ValueNumber或ValueDate字段中为这些字段创建值

例如:

*Customer*
1,BOB
2,JIM

*FieldDef*
1,Mobile,T
1,DateOfBirth,D

*CustomerField*
ID,CustomerID,FieldDefID,CaptureDate,ValueText,ValueNumber,ValueDate
1,1,1,2011-01-1,07123456789,NULL,NULL
2,1,2,2011-01-1,NULL,NULL,09-DEC-1980
3,1,1,2011-01-2,07123498787,NULL,NULL
我需要创建一个如下所示的视图:

*CustomerView*
ID,Name,Mobile,DateOfBirth
1,BOB,07123498787,09-DEC-1980
请注意,Bob的手机是列表中的第二个,因为它使用最新的捕获日期

理想情况下,我需要它是可扩展的,因此如果我将来创建一个新的字段def,它将在CustomerView中自动拾取

这在T-SQL中可能吗

谢谢


Simon。

您需要构建一个交叉表,这是您在TSQL中使用Pivot语句所做的。这里有一篇文章讨论如何动态构建pivot


这在视图中是不可能的,除非每次
FieldDef
更改时动态地重新创建视图,因为视图架构在创建时被锁定。但是,存储过程也可以这样做,这取决于您使用它的方式

编辑1

下面是一个仅适用于当前字段名的示例查询,通常需要通过动态SQL进行修改:

编辑2

修改为从customer字段表中获取最新值

with CustomerFieldNewest as (
    select
        cf1.*
    from
        customerfield cf1
    inner join
        (
            select
                customerid,
                fielddefid,
                max(capturedate) as maxcapturedate
            from
                customerfield cf2
            group by
                customerid,
                fielddefid
        ) cf2 on cf1.customerid = cf2.customerid
            and cf1.fielddefid = cf2.fielddefid
            and cf1.capturedate = cf2.maxcapturedate
)
,CustomerFieldPivot as (
    select
        C.ID as ID
        ,max(case when F.Name = 'Mobile' then CF.ValueText end) as Mobile
        ,max(case when F.Name = 'DateOfBirth' then CF.ValueDate end) as DateOfBirth
    from
        Customer C
    left join
        CustomerFieldNewest CF on C.ID = CF.CustomerID
    left join
        FieldDef F on F.ID = CF.FieldDefID
    group by
        C.ID
)
select
    C.*
    ,P.Mobile
    ,P.DateOfBirth
from
    Customer C
left join
    CustomerFieldPivot P on C.ID = P.ID
编辑3

下面是基于FieldDef中当前字段集动态生成视图的T-SQL代码(假设视图
CustomerView
已经存在,因此您需要首先将其创建为空白定义,否则会出现错误)。我不确定这一切的性能,但它应该正常工作

declare @sql varchar(max)
declare @fielddef varchar(max)
declare @fieldlist varchar(max)

select
    @fielddef = coalesce(@fielddef + ', ' + CHAR(13) + CHAR(10), '') +
        '           max(case when F.Name = ''' + F.Name + ''' then CF.' +
            case F.FieldType
                when 'T' then 'ValueText'
                when 'N' then 'ValueNumber'
                when 'D' then 'ValueDate'
            end
        + ' end) as [' + F.Name + ']'

    ,@fieldlist = coalesce(@fieldlist + ', ' + CHAR(13) + CHAR(10), '') +
        '       [' + F.Name + ']'
from
    FieldDef F

set @sql = '
    alter view [CustomerView] as

    with CustomerFieldNewest as (
        select
            cf1.*
        from
            customerfield cf1
        inner join
            (
                select
                    customerid,
                    fielddefid,
                    max(capturedate) as maxcapturedate
                from
                    customerfield cf2
                group by
                    customerid,
                    fielddefid
            ) cf2 on cf1.customerid = cf2.customerid
                and cf1.fielddefid = cf2.fielddefid
                and cf1.capturedate = cf2.maxcapturedate
    )
    ,CustomerFieldPivot as (
        select
            C.ID as ID,
' + @fielddef + '
        from
            Customer C
        left join
            CustomerFieldNewest CF on C.ID = CF.CustomerID
        left join
            FieldDef F on F.ID = CF.FieldDefID
        group by
            C.ID
    )
    select
        C.*,
' + @fieldlist + '
    from
        Customer C
    left join
        CustomerFieldPivot P on C.ID = P.ID
'

print @sql
exec(@sql)

select * from CustomerView

为了完整起见,有:


请按照页面左侧的格式指南来修改问题的格式。好的,谢谢。问题是该表可能包含多达2000万条客户记录。所以我不想每次都创建一个SP来拖网。有没有办法对SP进行编码以避免这种情况?哇!看起来不错。我会测试一下,然后回来找你。每次创建新字段定义时重新创建查询不是问题。还有一件事。。。max()函数是按字母顺序给我最新的手机号码还是最大的?我需要最新的基于CaptureDate@Simon:我修改为仅根据CaptureDate筛选出最新的customer字段值。@Simon:我添加了一个代码示例来动态创建视图。它假定该视图已经存在,因此在运行之前需要创建一个空视图
CustomerView
,并且它应该保持更新。不确定所有这些的性能,它可能会很慢。你的客户字段表索引好吗?谢谢Jeff,我来看看这篇文章。
 declare @t table (typ varchar(1), yuk sql_variant)
   insert @t values ('d', getdate())
   insert @t values ('i', 1234)
   insert @t values ('s', 'bleep bloop')

 select
    yuk, 
    case typ 
        when 'd' then convert(datetime, yuk, 106)+50
        when 'i' then cast(yuk as int) * 2
        when 's' then reverse(cast(yuk as varchar))
        else yuk
    end
 from @t