是否有可能在sql-server 2008中有一个视图将一行拆分为多个?

时间:2015-04-28 20:28:23

标签: sql-server sql-server-2008 views

后面的故事是我正在尝试编写一个视图,该视图采用一个表,每行都是一个ID,并在clob中为该ID提供序列化数据,并以sql可导航的形式呈现。基本上我的代码看起来像:

CREATE VIEW UNSERIALIZED_TABLE_VIEW AS 
SELECT
    SOURCE_TABLE.ID SOURCE_ID,
    a.*
FROM
    SOURCE_TABLE,
    FUNCTION_WHICH_UNSERIALIZES((SELECT DATA FROM SOURCE_TABLE WHERE ID = SOURCE_ID)

我尝试将该函数放在select语句中,但这只是一个语法错误,它是未定义的。当它运行时,错误通常是关于返回太多值的子查询。我可以批量反序列化数据,但现在我真的很好奇出了什么问题。

示例数据

@0History:23:ALPHANUMERICSTUFF1234567ID:11:ACCT1234567SourceMode:6:ANNUAL.ModeChanges:UniqueIndex:23:ALPHANUMERICSTUFF1234567ID:11:ACCT1234567OldValue:1:+NewValue:6:ANNUALChangeType:1:AChangeDate:20:6/03/2013 2:49:32 AM.
@0History:UniqueIndex:95:NOTTHESAME0987654|ALPHANUMERIC534|PRETEND349235|95CHARACTERSID:47:GNR44718500|PNR48CDQ704|PGP48090798|FGDS2345236SourceMode:26:ANNUAL|C-P-D|ANNUAL|ANNUALLoan:3:|||ModeChanges:UniqueIndex:95:00487SOMETHING4264500ORD|992581PROBABLY04ORD|0048SHOULD238BET|0095CHARS436PR638FGP07VDCID:47:GNR44718500|PNR48CDQ704|PGP48090798|FGDS2345236OldValue:7:+|+|+|+NewValue:26:ANNUAL|C-P-D|ANNUAL|ANNUALChangeType:7:A|A|A|AChangeDate:91:12/22/2013 11:53:11 PM|4/22/2013 11:53:11 PM|12/22/2013 11:53:11 PM|12/22/2013 11:53:11 PM.

数据是COLUMN_NAME格式的序列化表格数据:LENGTH_OF_ENTRY:DATA_FOR_COLUMN_ROW_1 | DATA_FOR_COLUMN_ROW2 | .... NEXT_COLUMN_NAME ......

功能示例:

CREATE FUNCTION FUNCTION_THAT_UNSERIALIZES (@clob varchar(max),@colname varchar(max)) RETURNS @NewValue TABLE (ID INT,value varchar(max)) AS
BEGIN
    DECLARE @colstart INT,@lenstart INT,@lenend INT,@collen VARCHAR(MAX),@lngth INT,@tmp VARCHAR(MAX), @rowid INT,@value VARCHAR(max),@next INT;
SELECT 
   @colstart = CHARINDEX(@colname,@tmp)+1,
   @lenstart = CHARINDEX(':',@tmp,@colstart)+1,
   @lenend   = CHARINDEX(':',@tmp,@lenstart),
   @collen   = SUBSTRING(@tmp,@lenstart,@lenend - @lenstart),
   @lngth    = CAST (@collen AS INT),
   @tmp      = SUBSTRING(@tmp,@lenend,@lngth);
WHILE LEN(@tmp) > 0 BEGIN
    SET @next = CHARINDEX('|',@tmp);
    IF @next > 0 BEGIN
        SET @value = SUBSTRING(@tmp,0,@next);     
        SET @tmp   = SUBSTRING(@tmp,@next+1,LEN(@tmp) - @next);
    END ELSE BEGIN
        SET @value = @tmp;    
        SET @tmp   = '';
 END
 INSERT INTO @NewValue VALUES(@rowid,@value)
 SET @rowid = @rowid+1;
 END
 RETURN 

示例错误

Msg 512, Level 16, State 1, Line 7

子查询返回的值超过1。当子查询遵循=,!=,<,< =,>,> =或子查询用作表达式时,不允许这样做。

Msg 4104, Level 16, State 1, Line 15

无法绑定多部分标识符“SOURCE_TABLE.SOURCE_ID”。

..我想可能还有另外一个,但无法弄清楚如何在这一分钟重现它。

1 个答案:

答案 0 :(得分:1)

我认为这可能是您完成我认为您要做的事情所需的语法。

CREATE VIEW UNSERIALIZED_TABLE_VIEW AS 
SELECT
    SOURCE_TABLE.ID SOURCE_ID,
    a.*
FROM SOURCE_TABLE
CROSS APPLY FUNCTION_WHICH_UNSERIALIZES(DATA, @colname) a

我不确定你的@colname参数应该是什么;它在问题中被遗漏了。