如何在SQL Server中替换2个特定字符之间的任何字符

2022-04-09 00:00:00 sql replace wildcard sql-server

我正在尝试替换包含这两个字符倍数的字符串中介于两个特定字符之间的任何字符。将其视为CSV格式。

下面是我在该字段中获得的数据的示例:

0001, ABCD1234;0002, EFGH432562;0003, IJKL1345hsth;...

我需要从其中检索的是‘,’之前的所有部分,而不是‘,’和‘;’之间的部分

我尝试了这些公式,但没有成功

 SELECT REPLACE(fieldname, ',[A-Z];', ' ') FROM ...
 or
 SELECT REPLACE(fieldname, ',*;', ' ') FROM ...

我需要获取

0001 0002 0003

有没有办法做到这一点?


解决方案

您可以CROSS APPLYSTRING_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;

相关文章