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

标签 t-sql view

我有一个正在努力实现的要求。如果可能的话,我希望使用 native 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 OR 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

我需要创建一个如下所示的 View :

*CustomerView*
ID,Name,Mobile,DateOfBirth
1,BOB,07123498787,09-DEC-1980

请注意,Bob 的手机是列表中的第二个手机,因为它使用最近的捕获日期。

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

这在 T-SQL 中可能吗?

谢谢

西蒙。

最佳答案

这对于 View 来说是不可能的,除非每次 FieldDef 更改时都会动态重新创建 View ,因为 View 架构在创建时被锁定。但是,使用存储过程可能可行,也可能不起作用,具体取决于您使用它的方式。

编辑 1

下面是一个示例查询,仅适用于您当前的字段名称,并且必须通过动态 SQL 进行修改才能正常工作:

编辑2

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

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

这里是 T-SQL 代码,用于根据 FieldDef 中当前的字段集动态生成 View (这假设 View 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

关于t-sql - 动态字段定义 - 这可以在 T-SQL 中完成吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5819338/

相关文章:

mysql - 重写此 SQL 查询而不使用 HAVING 子句?

sql - AdventureWorks 数据库和 dbo 前缀

sql - isnull 与 select case 语句的性能对比

sql-server - T-SQL (Azure),如何查看多个值是否在一个集合中

SQLite:当整行不唯一时删除重复行

php - MySQL:无法在 MySQL View 中插入

ios - 模态视图 Controller 中的文本不会在 View Controller 中更改

mysql - 无法创建包含联合的 View

sql - 在桥/数据相交表中查找 "missing"记录

sql - 查询考试成绩计算