如何在SQL Server中替换2个特定字符之间的任何字符
我正在尝试替换包含这两个字符倍数的字符串中介于两个特定字符之间的任何字符。将其视为CSV格式。
下面是我在该字段中获得的数据的示例:
0001, ABCD1234;0002, EFGH432562;0003, IJKL1345hsth;...
我需要从其中检索的是‘,’之前的所有部分,而不是‘,’和‘;’之间的部分
我尝试了这些公式,但没有成功
SELECT REPLACE(fieldname, ',[A-Z];', ' ') FROM ...
or
SELECT REPLACE(fieldname, ',*;', ' ') FROM ...
我需要获取
0001 0002 0003
有没有办法做到这一点?
解决方案
您可以CROSS APPLY
STRING_SPLIT
使用STRING_AGG(自SQL Server2017起)将数字粘合在一起。
select id, codes from your_table cross apply ( select string_agg(left(value, patindex('%_,%', value)), ' ') as codes from string_split(fieldname, ';') s where value like '%_,%' ) ca; GO
id | 代码 |
---|---|
1 | 0001 0002 0003 |
db;&>小提琴here
演示额外
以下是也可以在SQL Server 2014中使用的版本。
灵感来自@AaronBertrand的研究
UDF使用递归CTE拆分字符串。
而FOR XML
技巧是用来把数字粘在一起的。
CREATE FUNCTION dbo.fnString_Split ( @str nvarchar(4000), @delim nchar(1) ) RETURNS TABLE WITH SCHEMABINDING AS RETURN ( WITH RCTE AS ( SELECT 1 AS ordinal , ISNULL(NULLIF(CHARINDEX(@delim, @str),0), LEN(@str)) AS pos , LEFT(@str, ISNULL(NULLIF(CHARINDEX(@delim, @str),0)-1, LEN(@str))) AS value UNION ALL SELECT ordinal+1 , ISNULL(NULLIF(CHARINDEX(@delim, @str, pos+1), 0), LEN(@str)) , SUBSTRING(@str, pos+1, ISNULL(NULLIF(CHARINDEX(@delim, @str, pos+1),0)-pos-1, LEN(@str)-pos )) FROM RCTE WHERE pos < LEN(@str) ) SELECT ordinal, value FROM RCTE );
SELECT id, codes FROM your_table CROSS APPLY ( SELECT RTRIM(( SELECT LEFT(value, PATINDEX('%_,%', value))+' ' FROM dbo.fnString_Split(fieldname, ';') AS spl WHERE value LIKE '%_,%' ORDER BY ordinal FOR XML PATH(''), TYPE).value(N'./text()[1]', N'nvarchar(max)') ) AS codes ) ca OPTION (MAXRECURSION 250);
id | 代码 |
---|---|
1 | 0001 0002 0003 |
db;&>小提琴here
演示UDF的替代版本(无递归)
CREATE FUNCTION dbo.fnString_Split
(
@str NVARCHAR(4000),
@delim NCHAR(1)
)
RETURNS @tbl TABLE (ordinal INT, value NVARCHAR(4000))
WITH SCHEMABINDING
AS
BEGIN
DECLARE @value NVARCHAR(4000)
, @pos INT = 0
, @ordinal INT = 0;
WHILE (LEN(@str) > 0)
BEGIN
SET @ordinal += 1;
SET @pos = ISNULL(NULLIF(CHARINDEX(@delim, @str),0), LEN(@str)+1);
SET @value = LEFT(@str, @pos-1);
SET @str = SUBSTRING(@str, @pos+1, LEN(@str));
INSERT INTO @tbl (ordinal, value)
VALUES (@ordinal, @value);
END;
RETURN;
END;
相关文章