Динамическое извлечение нескольких текстов из одной строки - SQL - PullRequest
0 голосов
/ 14 февраля 2019

Значения в таблице представлены в следующем формате

Returns Request for Order: 121212, RA Number:1654356, 4 Products:, 1 x "ABC" (11169595), Expected Action: Replace, 1 x "FGH" (11449119), Expected Action: Refund, 1 x "XCV" (11100873), Expected Action: Refund, 1 x "VVV" (11028561), Expected Action: Refund 
-------------------------------------------------------------------------------------------------------------------------------
Returns Request for Order: 234354, RA Number:3465646, 3 Products:, 3 x "FGH" (11449119), Expected Action: Replace 
-----------------------------------------------------------------------------------------------------
Returns Request for Order: 234355, RA Number:3465646, 3 Products:, 2 x "ABC" (11169595),  Expected Action: Replace, 1 x "FGH" (11449119), Expected Action: Refund 

Ниже приводится требуемый формат o / p:

 Order_Num        REFUND_PRODUCTS                 REPLACE_PRODUCTS
 121212     11449119,11100873,11028561           11169595
 234354                                  11449119,11449119,11449119
 234355         11449119                       11169595,11169595

Если его '2 x "ABC" (11169595) Ожидаемое действие: REPLACE ', тогда в столбце REPLACE_PRODUCT число 11169595' должно появиться дважды.

если его «3 х ....», то трижды.Аналогично для REFUND_PRODUCT

Как я могу это сделать?

1 Ответ

0 голосов
/ 14 февраля 2019

ОК, пока ваш формат остается согласованным, вы можете использовать подстроки и модифицированную версию найденной функции здесь .Я создал быструю таблицу подсчета здесь, но вы можете поменять ее, если у вас уже есть таблица подсчета или чисел для использования.

Как Шон упомянул в комментариях, если это большая таблица, это происходитзанять время, чтобы бежать.Я не думаю, что на самом деле есть какой-то способ обойти это, хотя это упражнение по измельчению строк.

Declare @mytable as table (mydata varchar(4000))

Declare @myresults as table (order_num int,  refund_products varchar(1000), replace_products varchar(1000))

declare @tally as table (t_idx int)
insert into @tally
select 1 UNION ALL
select 2 UNION ALL
select 3 UNION ALL
select 4 UNION ALL
select 5 UNION ALL
select 6 UNION ALL
select 7 UNION ALL
select 8 UNION ALL
select 9

INSERT INTO @mytable
Select 'Returns Request for Order: 121212 RA Number:1654356 4 Products: 1 x "ABC" (11169595) Expected Action: REPLACE 1 x "FGH" (11449119) Expected Action: REFUND 1 x "XCV" (11100873) Expected Action: REFUND 1 x "VVV" (11028561) Expected Action: REFUND' UNION ALL
Select 'Returns Request for Order: 234354 RA Number:3465646 3 Products: 3 x "FGH" (11449119) Expected Action: REPLACE' UNION ALL
Select 'Returns Request for Order: 234355 RA Number:3465646 3 Products: 2 x "ABC" (11169595)  Expected Action: REPLACE 1 x "FGH" (11449119) Expected Action: REFUND'

insert into @myresults
Select
substring(mydata,charindex('Order:',mydata, 0)+7,6),
substring(
        STUFF(
            (Select ',' + pos_string from dbo.FindPatternLocation(mydata,'(') FOR XML PATH('')),1,1,'' ) ,10,500),

STUFF(
        (Select ',' + substring(mydata ,charindex('(',mydata,0)+1,8) from @tally where t_idx<=substring(mydata,charindex('(',mydata,0)-10,1) FOR XML PATH('')),1,1,''
        )

from @mytable

Select * from @myresults

Функция:

CREATE FUNCTION dbo.FindPatternLocation
(
    @string NVARCHAR(MAX),
    @term   NVARCHAR(255)
)
RETURNS TABLE
AS
    RETURN 
    (
      SELECT pos = Number - LEN(@term) ,

      substring(@string, Number - LEN(@term)+1, 8) as pos_string

      FROM (SELECT Number, Item = LTRIM(RTRIM(SUBSTRING(@string, Number, 
      CHARINDEX(@term, @string + @term, Number) - Number)))
      FROM (SELECT ROW_NUMBER() OVER (ORDER BY [object_id])
      FROM sys.all_objects) AS n(Number)
      WHERE Number > 1 AND Number <= CONVERT(INT, LEN(@string)+1)
      AND SUBSTRING(@term + @string, Number, LEN(@term)) = @term
    ) AS y);

В качестве альтернативы функцииВы можете использовать это как второй столбец: -

substring(
        STUFF(
            (Select ',' + pos_string from 

             (SELECT pos = Number - LEN('(') ,
              substring(mydata, Number - LEN('(')+1, 8) as pos_string

      FROM (SELECT Number, Item = LTRIM(RTRIM(SUBSTRING(mydata, Number, 
      CHARINDEX('(', mydata + '(', Number) - Number)))
      FROM (SELECT ROW_NUMBER() OVER (ORDER BY [object_id])
      FROM sys.all_objects) AS n(Number)
      WHERE Number > 1 AND Number <= CONVERT(INT, LEN(mydata)+1)
      AND SUBSTRING('(' + mydata, Number, LEN('(')) = '('
    ) aa) bb  FOR XML PATH('')),1,1,'' ) ,10,500)
...