SQL:根据第二个表替换多行的部分列

时间:2011-08-26 16:21:31

标签: sql sql-server-2008 replace multiple-tables

我有一个表(pages),其中包含网页的属性,包括页面内容本身的列,NVARCHAR(MAX)列。

在该列中,我需要查找并替换一堆文本字符串并将其替换为其他文本字符串;这些相关性位于第二个表格中{moving),其中包含oldValuenewValue列。

所以,例如,如果我从这两个表开始:

页面表:

ID Content
1  Words words Ancient words
2  Blah blah OutWithTheOld blah
3  Etc etc Useless etc

移动表:

OldValue          NewValue
Ancient           Better
OutWithTheOld     InWithTheNew
Useless           Useful

...我需要一种方法来制作一个像这样离开页面表的替换:

ID Content
1  Words words Better words
2  Blah blah InWithTheNew blah
3  Etc etc Useful etc

页面表中的给定记录可能需要多次替换,并且无法预测页面记录是否具有无,一个或许多必要的替换,或者来自moving.oldvalue的值将是找到并需要更换。

我正在使用SQL Server 2008,而且我对它很新。非常感谢您提供的任何帮助!

2 个答案:

答案 0 :(得分:3)

这是一个使用CTE的单语句非游标方法:

WITH CTE(iteration, page_id, content) AS (
    SELECT
        0,
        P.page_id,
        REPLACE(P.content, M1.old_value, M1.new_value)
    FROM
        Pages P
    INNER JOIN Moving M1 ON
        P.content LIKE '%' + M1.old_value + '%'
    WHERE
        NOT EXISTS (SELECT * FROM Moving M2 WHERE P.content LIKE '%' + M2.old_value + '%' AND M2.moving_id < M1.moving_id)
    UNION ALL
    SELECT
        CTE.iteration + 1,
        CTE.page_id,
        REPLACE(CTE.content, M3.old_value, M3.new_value)
    FROM
        CTE
    INNER JOIN Moving M3 ON
        CTE.content LIKE '%' + M3.old_value + '%'
    WHERE
        NOT EXISTS (SELECT * FROM Moving M4 WHERE CTE.content LIKE '%' + M4.old_value + '%' AND M4.moving_id < M3.moving_id)
)
UPDATE P2
SET
    content = CTE1.content
FROM
    Pages P2
INNER JOIN CTE CTE1 ON
    CTE1.page_id = P2.page_id AND
    NOT EXISTS (SELECT * FROM CTE CTE2 WHERE page_id = P2.page_id AND CTE2.iteration > CTE1.iteration)

答案 1 :(得分:2)

试试这个

DECLARE @OldV NVARCHAR(32)    -- Adjust for your field sizes in MOVING
DECLARE @NEWV NVARCHAR(32)

DECLARE db_cursor CURSOR FOR  
SELECT *
FROM Moving

OPEN db_cursor   
FETCH NEXT FROM db_cursor INTO @OldV,@newV

WHILE @@FETCH_STATUS = 0   
BEGIN   
    UPDATE Pages SET content=REPLACE(content,@oldV,@NewV)
      WHERE content LIKE '%'+@OldV+'%'
       FETCH NEXT FROM db_cursor INTO @oldV,@NewV
END   

CLOSE db_cursor   
DEALLOCATE db_cursor

虽然我一般不喜欢游标,但这应该可以帮到你了