Need a query help
Need a query help. I have a table where data is represented like this
userid TransactionTime FunctionKey
1018 2011-07-27 09:14:38.000 0
1018 2011-07-27 09:17:11.000 99
1018 2011-07-27 09:38:22.000 0
1018 2011-07-27 10:34:50.000 99
I need a query which will return a dataset like this:
userid Intime Outtime
101开发者_运维知识库8 2011-07-27 09:14:38.000 2011-07-27 09:17:11.000
1018 2011-07-27 09:38:22.000 2011-07-27 10:34:50.000
Please Help..
declare @T table (userid int, TransactionTime datetime, FunctionKey int)
insert into @T values
(1018, '2011-07-27 09:14:38.000', 0),
(1018, '2011-07-27 09:17:11.000', 99),
(1018, '2011-07-27 09:38:22.000', 0),
(1018, '2011-07-27 10:34:50.000', 99),
(1019, '2011-07-27 09:14:38.000', 0),
(1019, '2011-07-27 09:17:11.000', 0),
(1019, '2011-07-27 09:38:22.000', 99),
(1020, '2011-07-27 09:14:38.000', 0),
(1020, '2011-07-27 09:17:11.000', 99),
(1020, '2011-07-27 09:38:22.000', 99)
;with cte as
(
select userid,
TransactionTime,
FunctionKey,
row_number() over(partition by userid order by TransactionTime) as rn
from @T
)
select C1.userid,
C1.TransactionTime as InTime,
C2.TransactionTime as OutTime
from cte as C1
left outer join cte as C2
on C1.userid = C2.userid and
C1.rn + 1 = C2.rn
where C1.FunctionKey = 0
Result:
userid InTime OutTime
----------- ----------------------- -----------------------
1018 2011-07-27 09:14:38.000 2011-07-27 09:17:11.000
1018 2011-07-27 09:38:22.000 2011-07-27 10:34:50.000
1019 2011-07-27 09:14:38.000 2011-07-27 09:17:11.000
1019 2011-07-27 09:17:11.000 2011-07-27 09:38:22.000
1020 2011-07-27 09:14:38.000 2011-07-27 09:17:11.000
WITH
ordered AS
(
SELECT
ROW_NUNBER() OVER (PARTITION BY userid ORDER BY TransactionTime, FunctionKey) item_id,
*
FROM
myTable
)
, logins AS
(
SELECT * FROM ordered WHERE FunctionKey = 0
)
, logouts AS
(
SELECT * FROM ordered WHERE FunctionKey = 99
)
SELECT
COALESCE([logins].userid, [logouts].userid),
[logins].TransactionTime,
[logouts].TransactionTime
FROM
[logins]
FULL OUTER JOIN
[logouts]
ON [logouts].userid = [logins].userid
AND [logouts].item_id = [logins].item_id + 1
AND [logouts].FunctionKey <> [logins].FunctionKey
ORDER BY
COALESCE([logins].TransactionTime, [logouts].TransactionTime)
Alternatively, this one deals with sequences of logins by taking the first login, and sequences of logogut by taking the last logout.
WITH
ordered AS
(
SELECT
ROW_NUNBER() OVER (PARTITION BY userid ORDER BY TransactionTime, FunctionKey)
-
ROW_NUMBER() OVER (PARTITION BY userid, FunctionKey ORDER BY TransactionTime)
AS group_id,
*
FROM
myTable
)
,
filtered AS
(
SELECT
userid,
FunctionKey,
CASE WHEN FunctionKey = 0 THEN MIN(TransactionTime) ELSE MAX(TransactionTime) END AS TransactionTime
FROM
ordered
GROUP BY
userid,
FunctionKey,
group_id
)
,
sequenced AS
(
SELECT
ROW_NUMBER() OVER (PARTITION BY userid ORDER BY TransactionTime, FunctionKey) AS item_id,
*
FROM
filtered
)
, logins AS
(
SELECT * FROM sequenced WHERE FunctionKey = 0
)
, logouts AS
(
SELECT * FROM sequenced WHERE FunctionKey = 99
)
SELECT
COALESCE([logins].userid, [logouts].userid),
[logins].TransactionTime,
[logouts].TransactionTime
FROM
[logins]
FULL OUTER JOIN
[logouts]
ON [logouts].userid = [logins].userid
AND [logouts].item_id = [logins].item_id + 1
AND [logouts].FunctionKey <> [logins].FunctionKey
ORDER BY
COALESCE([logins].TransactionTime, [logouts].TransactionTime)
Join the table to itself, taking one timestamp from each:
select
t1.userid,
t1.TransactionTime as InTime
t2.TransactionTime as Outtime
from mytable t1
join mytable t2 on t2.userid = t1.userid and t2.FunctionKey = 99
where t1.FunctionKey = 0;
精彩评论