tsql因为没有数组的概念,那么如何把一批数据传给数据库引擎让它一次性处理完呢。那么我们可以在客户端把数据用逗号串联起来,然后把这些数据作为NVARCHAR(MAX)传给数据库,然后在数据库分割这个用逗号分隔的字符串。
先不谈怎么分割这个有逗号分隔的字符串,看看怎么把数据库中的记录用逗号分隔检索出来,如下有一种方式:
DECLARE @vv AS NVARCHAR(MAX) ;With employees(ID, Name) AS( SELECT 1, 'Eric' UNION ALL SELECT 2, 'Petter' UNION ALL SELECT 3, 'Wang' UNION ALL SELECT 4, 'Jeff' UNION ALL SELECT 5, NULL ) SELECT @vv = COALESCE(@vv + ', ', '') + CAST(Name AS NVARCHAR(10)) FROM employees WHERE Name IS NOT NULL SELECT @vv
结果如下:
Eric, Petter, Wang, Jeff
现在来看看如何分割字符串(其实就是分割CSV字符串):
第一种方式,找到逗号所在位置,然后提出逗号前的部分,然后循环,处理第二个逗号。。。代码如下,很简单:
CREATE FUNCTION [dbo].[parseCsv1](@chars NVARCHAR(MAX)) RETURNS @tv TABLE(value NVARCHAR(50)) AS BEGIN DECLARE @spIndex AS INT DECLARE @varItem AS NVARCHAR(MAX) SET @chars = @chars + ',' WHILE PATINDEX('%,%', @chars) <> 0 BEGIN SET @spIndex = PATINDEX('%,%', @chars) SET @varItem = LEFT(@chars, @spIndex-1) INSERT @tv VALUES(@varItem) SET @chars = STUFF(@chars, 1, @spIndex, '') END RETURN END
第二种方式,使用集合的方式来处理,代码如下:
CREATE FUNCTION parseCsv2(@varChars NVARCHAR(MAX)) RETURNS @tv TABLE(value NVARCHAR(50)) AS BEGIN SET @varChars = ',' + @varChars + ',' ;WITH cte1 AS( SELECT N sI FROM dbo.Number WHERE N <= LEN(@varChars)AND SUBSTRING(@varChars, N, 1) =','), cte2 (indStart, indEnd) AS(SELECT c.* , c2.* FROM cte1 c CROSS APPLY( SELECT TOP 1 * FROM cte1 WHERE si > c.si) as c2) INSERT INTO @tv SELECT V FROM( SELECT *, RTRIM(LTRIM(SUBSTRING(@varChars, indStart+1, indEnd -indStart-1))) AS v FROM cte2 ) AS d1 WHERE v <> '' RETURN END
这种方式完全没有使用到循环,采用了集合的方式来处理字符串。