从备忘录SQL中解析数据

时间:2017-10-05 21:35:05

标签: sql sql-server tsql string-parsing

所以基本上我在DB中有几个备忘录。我需要通过它们进行查询,并且只抓取备忘录中包含日期之前的每个备注。所以,如果我有这个例子:

abc def ghi jkl 9/1/17: mno pqr

我只想抓住9/1/17: mno pqr部分

有些人甚至在第一个初始日期之后有更多的条目,如下所示:

abc def ghi jkl 9/1/17: mno pqr 9/2/17: stu vwx yz

在这种情况下,我需要9/1/17: mno pqr9/2/17: stu vwx yz

可能会将每个值放在一个新行中,如下所示:

1: 9/1/17: mno pqr 2: 9/2/17: stu vwx yz

这样如果它出现故障我可以根据需要重新排序。

任何帮助将不胜感激!感谢

2 个答案:

答案 0 :(得分:0)

您可以使用charindex或patindex

假设, 1.您的文字不包含' /'除了在日期 2.您的日期将分隔符设置为' /'

以下查询将返回第一次出现的' /'在你的字符串中。

SELECT PATINDEX('%/%', yourcolumn) 
  FROM yourtable

从上述查询返回的值 - 2将是您日期的起始位置。整个字符串的长度 - 从上面的查询返回的值将是输出的长度。

SELECT susbtring(yourcolumn, PATINDEX('%/%', yourcolumn) -2, 
         length(yourcolumn)-PATINDEX('%/%', yourcolumn))
  FROM yourtable

如果您要处理多个日期,则必须使用递归CTE

答案 1 :(得分:0)

您可以使用PatternSplitCM(DDL用于以下功能)。解决方案看起来像这样(请注意,您需要运行SQL Server 2012+,因为我使用LEAD):

declare @string varchar(255) = 
'abcdefg 9/21/17 took the notes. 9/23/17 printed the notes. 9/21/17 took the notes. 9/23/17 printed the notes.'

select ItemNumber = concat(ItemNumber/2,':'), Item
from 
(
  select ItemNumber, item = item +' '+ LEAD(item, 1) OVER (ORDER BY itemNumber), [Matched]
  from dbo.PatternSplitCM(@string, '[0-9/]')
) ExtractDates
where [Matched] = 1;

<强>结果

ItemNumber   Item
------------ ----------------------------
1:           9/21/17  took the notes. 
2:           9/23/17  printed the notes. 
3:           9/21/17  took the notes. 
4:           9/23/17  printed the notes.

功能

-- PatternSplitCM will split a string based on a pattern of the form 
-- supported by LIKE and PATINDEX 
-- 
-- Created by: Chris Morris 12-Oct-2012 
ALTER FUNCTION [dbo].[PatternSplitCM]
(
       @List                VARCHAR(8000) = NULL
       ,@Pattern            VARCHAR(50)
) RETURNS TABLE WITH SCHEMABINDING 
AS    
RETURN
    WITH numbers AS (
      SELECT TOP(ISNULL(DATALENGTH(@List), 0))
       n = ROW_NUMBER() OVER(ORDER BY (SELECT NULL))
      FROM
      (VALUES (0),(0),(0),(0),(0),(0),(0),(0),(0),(0)) d (n),
      (VALUES (0),(0),(0),(0),(0),(0),(0),(0),(0),(0)) e (n),
      (VALUES (0),(0),(0),(0),(0),(0),(0),(0),(0),(0)) f (n),
      (VALUES (0),(0),(0),(0),(0),(0),(0),(0),(0),(0)) g (n))
    SELECT
      ItemNumber = ROW_NUMBER() OVER(ORDER BY MIN(n)),
      Item = SUBSTRING(@List,MIN(n),1+MAX(n)-MIN(n)),
      [Matched]
     FROM (
      SELECT n, y.[Matched], Grouper = n - ROW_NUMBER() OVER(ORDER BY y.[Matched],n)
      FROM numbers
      CROSS APPLY (
          SELECT [Matched] = CASE WHEN SUBSTRING(@List,n,1) LIKE @Pattern THEN 1 ELSE 0 END
      ) y
     ) d
     GROUP BY [Matched], Grouper;