sql - 使用 CTE 进行字符串拆分的有效方法

标签 sql sql-server sql-server-2005 tsql common-table-expression

我有一张看起来像的 table

ID  Layout
1   hello,world,welcome,to,tsql
2   welcome,to,stackoverflow

所需的输出应该是
Id  Splitdata
1   hello
1   world
1   welcome
1   to
1   tsql
2   welcome
2   to
2   stackoverflow

我已通过以下查询完成此操作
Declare @t TABLE(
    ID  INT IDENTITY PRIMARY KEY,
    Layout VARCHAR(MAX)
)
INSERT INTO @t(Layout)
SELECT 'hello,world,welcome,to,tsql' union all
SELECT 'welcome,to,stackoverflow'
--SELECT * FROM @t
;With cte AS(
select F1.id
 ,O.splitdata 
 from
 (
 select *,
 cast('<X>'+replace(F.Layout,',','</X><X>')+'</X>' as XML) as xmlfilter
 from @t F
 )F1
 cross apply
 ( 
 select fdata.D.value('.','varchar(MAX)') as splitdata 
 from f1.xmlfilter.nodes('X') as fdata(D)) O
 )

 select * from cte

但从性能上来说,这是非常糟糕的。我正在寻找更有效的查询,但仅使用 CTE。

最佳答案

你似乎死心塌地使用 CTE,所以试试这个:

DECLARE @YourTable table (RowID int, Layout varchar(200))
INSERT @YourTable VALUES (1,'hello,world,welcome,to,tsql')
INSERT @YourTable VALUES (2,'welcome,to,stackoverflow')

;WITH SplitSting AS
(
    SELECT
        RowID,LEFT(Layout,CHARINDEX(',',Layout)-1) AS Part
            ,RIGHT(Layout,LEN(Layout)-CHARINDEX(',',Layout)) AS Remainder
        FROM @YourTable
        WHERE Layout IS NOT NULL AND CHARINDEX(',',Layout)>0
    UNION ALL
    SELECT
        RowID,LEFT(Remainder,CHARINDEX(',',Remainder)-1)
            ,RIGHT(Remainder,LEN(Remainder)-CHARINDEX(',',Remainder))
        FROM SplitSting
        WHERE Remainder IS NOT NULL AND CHARINDEX(',',Remainder)>0
    UNION ALL
    SELECT
        RowID,Remainder,null
        FROM SplitSting
        WHERE Remainder IS NOT NULL AND CHARINDEX(',',Remainder)=0
)
SELECT * FROM SplitSting ORDER BY RowID

输出:
RowID       Part                   
----------- -----------------------
1           hello                  
1           world                  
1           welcome                
1           to                     
1           tsql                   
2           welcome                
2           to                     
2           stackoverflow          

(8 row(s) affected)

这是一篇关于在 SQL Server 中拆分字符串的优秀文章:"Arrays and Lists in SQL Server 2005 and Beyond, When Table Value Parameters Do Not Cut it" by Erland Sommarskog

编辑 这是另一个版本(但您需要一个数字表)返回与上述相同的结果:
;WITH SplitValues AS
(
    SELECT
        RowID,ListValue
        FROM (SELECT
                  RowID, LTRIM(RTRIM(SUBSTRING(List2, number+1, CHARINDEX(',', List2, number+1)-number - 1))) AS ListValue
                  FROM (
                           SELECT RowID, ',' + Layout + ',' AS List2
                           FROM @YourTable
                       ) AS dt
                      INNER JOIN Numbers n ON n.Number < LEN(dt.List2)
                  WHERE SUBSTRING(List2, number, 1) = ','
             ) dt2
        WHERE ListValue IS NOT NULL AND ListValue!=''
)
SELECT * FROM SplitValues

在这里查看数字表:What is the best way to create and populate a numbers table?

关于sql - 使用 CTE 进行字符串拆分的有效方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6381587/

相关文章:

mysql - ORDER BY CASE 行为不正确

sql - 如何在oracle程序内部传递date参数?

c# - MVC : second parameter for SqlCommand appearing as Null in Database

sql - 如何从 SQL Server 2005 中的表中删除锁定?

mysql - 如何合并两个MySQL表

sql - sql 'like' 查询中的字符代码?

sql-server - 如何连接到Docker容器内的SQL Server数据库?

c# - 为什么将我们的 .NET/SQL Server 网站移动到新主机会导致超过连接池大小?

SQL Server 2005 RIGHT OUTER JOIN 不起作用

sql-server-2005 - 带有动态参数的 SQL Server Reporting Services 订阅