Преобразование Rowdata в несколько строк на основе условия - PullRequest
0 голосов
/ 01 мая 2018

Я пытаюсь извлечь родительские и дочерние идентификаторы из одного столбца на основе даты

     Cust_id     ID       Date 

    75407014    603    2018-04-27 
    79807014    603    2018-04-30 
    75407016    604    2018-04-23 
    79807016    604    2018-04-30 
    75407018    605    2018-04-24 
    79807018    605    2018-04-30 
    75407020    606    2018-04-24 
    79807020    606    2018-04-30 
    75407014    608    2018-04-27 

Я хочу исключить id = 608, так как есть только одна строка, мой ожидаемый вывод относится к родительскому и дочернему идентификатору

Select Row_number () over (partition by Cust_id, id order by date ) rn 

Мой ожидаемый результат:

Parent_id  Child_id 
75407014   79807014
75407016   79807016
75407018   79807018
75407020   79807020

Итак, Cust_id на первую дату будет parent_id, а на следующую дату будет child_id.ID - это обычный столбец ссылок для идентификатора customer_id

Большое спасибо за вашу помощь

Ответы [ 5 ]

0 голосов
/ 01 мая 2018
           This query works if ID values repeat twice   
           SELECT * INTO TemP FROM
               (
                SELECT 75407014 Cust_id,    603 ID,   '2018-04-27'Date UNION
                SELECT 79807014 Cust_id,    603 ID,   '2018-04-30'Date UNION
                SELECT 75407016 Cust_id,    604 ID,   '2018-04-23'Date UNION
                SELECT 79807016 Cust_id,    604 ID,   '2018-04-30'Date UNION
                SELECT 75407018 Cust_id,    605 ID,   '2018-04-24'Date UNION
                SELECT 79807018 Cust_id,    605 ID,   '2018-04-30'Date UNION
                SELECT 75407020 Cust_id,    606 ID,   '2018-04-24'Date UNION
                SELECT 79807020 Cust_id,    606 ID,   '2018-04-30'Date UNION
                SELECT 75407014 Cust_id,    608 ID,   '2018-04-27'Date 
                )A

                SELECT *,ROW_number() OVER (PARTITION BY ID ORDER BY Date) RNO INTO #Tem
                FROM TEMP 

                SELECT K.Cust_id,L.cust_id
                FROM #Tem K
                INNER JOIN  #Tem L
                ON K.ID = L.ID AND K.RNO < L.RNO
0 голосов
/ 01 мая 2018

Пожалуйста, попробуйте это

;WITH CTE(Cust_id,ID,[Date ])
AS
(
SELECT 75407014,603,'2018-04-27' UNION ALL 
SELECT 79807014,603,'2018-04-30' UNION ALL 
SELECT 75407016,604,'2018-04-23' UNION ALL 
SELECT 79807016,604,'2018-04-30' UNION ALL 
SELECT 75407018,605,'2018-04-24' UNION ALL 
SELECT 79807018,605,'2018-04-30' UNION ALL 
SELECT 75407020,606,'2018-04-24' UNION ALL 
SELECT 79807020,606,'2018-04-30' UNION ALL 
SELECT 75407014,608,'2018-04-27' 
)
,CTE_Final
AS
(
SELECT Cust_id
      ,ID
      ,[Date ]
      ,DENSE_RANK()OVER(ORDER BY ID) SEQ 
FROM CTE 
)
SELECT Parent_id
      ,Child_id 
FROM
    (
    SELECT SUBSTRING(Cust_id,0,CHARINDEX(',',Cust_id))  AS Parent_id,
       SUBSTRING(Cust_id,CHARINDEX(',',Cust_id)+1,LEN(Cust_id))AS Child_id 
     FROM
    (
    SELECT DISTINCT STUFF((SELECT DISTINCT  ', '+CAST(  Cust_id AS VARCHAR(10))
    FROM CTE_Final i WHERE i.SEQ=o.SEQ 
    FOR XML PATH ('')),1,1,'') AS  Cust_id
    FROM CTE_Final o
    )dt
)Dt2
WHERE  Dt2.Parent_id <> ''
ORDER BY Parent_id

Результат

 Parent_id   Child_id
 --------------------
 75407014    79807014
 75407016    79807016
 75407018    79807018
 75407020    79807020
0 голосов
/ 01 мая 2018

Вы можете использовать lead с partition by id, чтобы получить желаемый результат.

Для SQL Server 2012 и выше

select * 
from   (select [cust_id], 
               lead([cust_id]) 
                 over(partition by id order by date) as Child_id 
        from   tablename t1)t 
where  child_id is not null 

DEMO

выход

+----------+----------+
| Cust_id  | Child_id |
+----------+----------+
| 75407014 | 79807014 |
+----------+----------+
| 75407016 | 79807016 |
+----------+----------+
| 75407018 | 79807018 |
+----------+----------+
| 75407020 | 79807020 |
+----------+----------+

Для более старой версии SQL Server вы можете использовать подзапрос, чтобы найти пример, как показано ниже.

select * 
from   (select [cust_id], 
               (select top 1 [cust_id] 
                from   tablename t2 
                where  t2.id = t1.id 
                       and t2.date > t1.date 
                order  by t2.date) as Child_id 
        from   tablename t1)t 
where  child_id is not null 
0 голосов
/ 01 мая 2018

На основании данных примера? Есть простой метод и более продвинутый метод.

См. Пример фрагмента ниже:

declare @TestTable table (Cust_id int, ID int, [Date] date);

insert into @TestTable (Cust_id, ID, [Date]) values 
(75407014, 603, '2018-04-27'), 
(79807014, 603, '2018-04-30'), 
(75407016, 604, '2018-04-23'), 
(79807016, 604, '2018-04-30'), 
(75407018, 605, '2018-04-24'), 
(79807018, 605, '2018-04-30'), 
(75407020, 606, '2018-04-24'), 
(79807020, 606, '2018-04-30'), 
(75407014, 608, '2018-04-27');

--
-- Method 1: A simple self-join on ID and Date
-- This assumes that there are maximum 2 records per ID
--
select t1.Cust_id as Parent_id, t2.Cust_id as Child_id 
from @TestTable t1
join @TestTable t2 on (t2.ID = t1.ID and t2.[Date] > t1.[Date]);

--
-- Method 2: use a CTE with a row_number and self-join the CTE
--
;WITH CTE as (
  select 
    ID, 
    row_number() over (partition by ID order by [Date], Cust_id) as rn, 
    Cust_id 
  from @TestTable t
)
select t1.Cust_id as Parent_id, t2.Cust_id as Child_id
from CTE t1
join CTE t2 on (t1.ID = t2.ID and t1.rn = 1 and t2.rn > 1);

Второй метод позволяет иметь более 1 ребенка на одного родителя.

0 голосов
/ 01 мая 2018

Вы можете использовать row_number() функцию

with t as (
 select *,
       row_number() over (partition by id order by date) Seq 
 from table t
 where exists (select 1 from table where id = t.id group by id having count(*) > 1)
)

select (case when seq = 1 then Cust_id end) Parent_id,
       (case when seq = 2 then Cust_id end) Child_id   
from t
where seq in (1,2);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...