IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[vGetVisits]') AND type in (N'U'))
DROP TABLE [dbo].[vGetVisits]
GO
CREATE TABLE [dbo].[vGetVisits](
[id] [int] NOT NULL,
[mydate] [datetime] NOT NULL,
CONSTRAINT [PK_vGetVisits] PRIMARY KEY CLUSTERED
(
[id] ASC
)
)
GO
INSERT INTO [dbo].[vGetVisits]([id], [mydate])
VALUES
(1, '2014-01-01 11:00'),
(2, '2014-01-03 10:00'),
(3, '2014-01-04 09:30'),
(4, '2014-04-01 10:00'),
(5, '2014-05-01 11:00'),
(6, '2014-07-01 09:00'),
(7, '2014-07-31 08:00');
GO
-- Clean up
IF OBJECT_ID (N'dbo.udfLastHitRecursive', N'FN') IS NOT NULL
DROP FUNCTION udfLastHitRecursive;
GO
-- Actual Function
CREATE FUNCTION dbo.udfLastHitRecursive
( @MyDate datetime)
RETURNS TINYINT
AS
BEGIN
-- Your returned value 1 or 0
DECLARE @Returned_Value TINYINT;
SET @Returned_Value=0;
-- Prepare gaps table to be used.
WITH gaps AS
(
-- Select Date and MaxDiff from the original table
SELECT
CONVERT(Date,mydate) AS [date]
, DATEDIFF(day,ISNULL(LAG(mydate, 1) OVER (ORDER BY mydate), mydate) , mydate) AS [MaxDiff]
FROM dbo.vGetVisits
)
SELECT @Returned_Value=
(SELECT DISTINCT -- DISTINCT in case we have same date but different time
CASE WHEN
(
-- It is a first entry
[date]=(SELECT MIN(CONVERT(Date,mydate)) FROM dbo.vGetVisits))
OR
/*
--Gap between last qualifying date and entered is greater than 90
Calculate Running sum upto and including required date
and find a remainder of division by 91.
*/
((SELECT SUM(t1.MaxDiff)
FROM (SELECT [MaxDiff] FROM gaps WHERE [date]<=t2.[date]
) t1
)%91 -
/*
ISNULL added to include first value that always returns NULL
Calculate Running sum upto and NOT including required date
and find a remainder of division by 91
*/
ISNULL((SELECT SUM(t1.MaxDiff)
FROM (SELECT [MaxDiff] FROM gaps WHERE [date]<t2.[date]
) t1
)%91, 0) -- End ISNULL
<0 )
/* End Running sum upto and including required date */
OR
-- Gap between two nearest dates is greater than 90
((SELECT SUM(t1.MaxDiff)
FROM (SELECT [MaxDiff] FROM gaps WHERE [date]<=t2.[date]
) t1
) - ISNULL((SELECT SUM(t1.MaxDiff)
FROM (SELECT [MaxDiff] FROM gaps WHERE [date]<t2.[date]
) t1
), 0) > 90)
THEN 1
ELSE 0
END
AS [Qualifying]
FROM gaps t2
WHERE [date]=CONVERT(Date,@MyDate))
-- What is neccesary to return when entered date is not in dbo.vGetVisits?
RETURN @Returned_Value
END
GO
SELECT
dbo.udfLastHitRecursive(mydate) AS [Qualifying]
, [id]
, mydate
FROM dbo.vGetVisits
ORDER BY mydate
ফলাফল
এসকিউএল সার্ভারে রানিং টোটাল কীভাবে গণনা করা যায় তা দেখুন
আপডেট: পারফরম্যান্স পরীক্ষার ফলাফল নীচে দেখুন।
"90 দিনের ব্যবধান" ইয়ারক्यूबের এবং আমার সমাধানগুলি অক্ষত রাখলে যদি পল হোয়াইটের সমাধানগুলিতে বিভিন্ন ফলাফল ফিরে আসতে পারে তবে সন্ধানের ক্ষেত্রে বিভিন্ন যুক্তি ব্যবহার করার কারণে। এটি যথাক্রমে DATEDIFF এবং DATEADD ফাংশন ব্যবহারের কারণে ।
উদাহরণ স্বরূপ:
SELECT DATEADD(DAY, 90, '2014-01-01 00:00:00.000')
'2014-04-01 00: 00: 00.000' অর্থ '2014-04-01 01: 00: 00.000' 90 দিনের ব্যবধান ছাড়িয়ে গেছে
কিন্তু
SELECT DATEDIFF(DAY, '2014-01-01 00:00:00.000', '2014-04-01 01:00:00.000')
'90' অর্থ ফেরৎ দেয় যে এটি এখনও ব্যবধানের মধ্যে রয়েছে।
একজন বিক্রেতার উদাহরণ বিবেচনা করুন। এই ক্ষেত্রে '2014-01-01' তারিখে '2014-01-01' 23: 59: 59: 999 'এ' 2014-01-01 'তারিখে বিক্রি হওয়া একটি পচনশীল পণ্য বিক্রি করা ঠিক আছে। এক্ষেত্রে DATEDIFF (DAY, ...) এর মান ঠিক আছে।
আরেকটি উদাহরণ হ'ল রোগীর দেখা হওয়ার অপেক্ষায়। যে কেউ '2014-01-01 00: 00: 00: 000' এ এসে '2014-01-01 23: 59: 59: 999' এ চলে যায় তার জন্য 0 (শূন্য) দিন হয় যদি ডেটাডেআইএফ ব্যবহার করা হয় তবে প্রকৃত অপেক্ষা প্রায় 24 ঘন্টা ছিল। আবার রোগী যা '2014-01-01 23:59:59' এ আসে এবং '2014-01-02 00:00:01' এ চলে যায় 'DATEDIFF ব্যবহার করা হয় তবে একদিনের জন্য অপেক্ষা করছিল।
কিন্তু আমার দ্বিমত আছে.
আমি DATEDIFF সমাধানগুলি ছেড়ে দিয়েছি এমনকি পারফরম্যান্সও সেগুলি পরীক্ষা করেছি তবে সেগুলি সত্যই তাদের নিজস্ব লিগে থাকা উচিত।
এছাড়াও এটি উল্লেখ করা হয়েছিল যে বড় ডেটাসেটগুলির জন্য একই দিনের মানগুলি এড়ানো সম্ভব নয়। সুতরাং যদি আমরা 2 মিলিয়ন ডেটা জুড়ে 13 মিলিয়ন রেকর্ডগুলি বলতে পারি তবে আমরা কিছু দিনের জন্য একাধিক রেকর্ড ধারণ করব। সেই রেকর্ডগুলি আমার এবং ইপারকিউবের ডেটেডিএফ সমাধানগুলিতে প্রাথমিক পর্যায়ে ফিল্টার করা হচ্ছে। আশা করি ইপারকিউব এতে কিছু মনে করবে না।
নিম্নলিখিত টেবিলে সমাধানগুলি পরীক্ষা করা হয়েছিল
CREATE TABLE [dbo].[vGetVisits](
[id] [int] NOT NULL,
[mydate] [datetime] NOT NULL,
)
দুটি পৃথক ক্লাস্টারযুক্ত সূচক (এই ক্ষেত্রে মাইডেট) সহ:
CREATE CLUSTERED INDEX CI_mydate on vGetVisits(mydate)
GO
সারণিটি নিম্নলিখিত উপায়ে জনবহুল ছিল
SET NOCOUNT ON
GO
INSERT INTO dbo.vGetVisits(id, mydate)
VALUES (1, '01/01/1800')
GO
DECLARE @i bigint
SET @i=2
DECLARE @MaxRows bigint
SET @MaxRows=13001
WHILE @i<@MaxRows
BEGIN
INSERT INTO dbo.vGetVisits(id, mydate)
VALUES (@i, DATEADD(day,FLOOR(RAND()*(3)),(SELECT MAX(mydate) FROM dbo.vGetVisits)))
SET @i=@i+1
END
একটি মিলিয়ন মিলিয়ন কেসের ক্ষেত্রে INSERT এমনভাবে পরিবর্তন করা হয়েছিল যে 0-20 মিনিটের এন্ট্রি এলোমেলোভাবে যুক্ত করা হয়েছিল।
সমস্ত সমাধানগুলি সাবধানতার সাথে নিম্নলিখিত কোডটিতে মোড়ানো হয়েছিল
SET NOCOUNT ON
GO
DECLARE @StartDate DATETIME
SET @StartDate = GETDATE()
--- Code goes here
PRINT 'Total milliseconds: ' + CONVERT(varchar, DATEDIFF(ms, @StartDate, GETDATE()))
প্রকৃত কোডগুলি পরীক্ষা করা হয়েছে (কোনও নির্দিষ্ট ক্রমে নয়):
ইপারকিউবের DATEDIFF সমাধান ( YPC, DATEDIFF )
DECLARE @cd TABLE
( TheDate datetime PRIMARY KEY,
Qualify INT NOT NULL
);
DECLARE
@TheDate DATETIME,
@Qualify INT = 0,
@PreviousCheckDate DATETIME = '1799-01-01 00:00:00'
DECLARE c CURSOR
LOCAL STATIC FORWARD_ONLY READ_ONLY
FOR
SELECT
mydate
FROM
(SELECT
RowNum = ROW_NUMBER() OVER(PARTITION BY cast(mydate as date) ORDER BY mydate)
, mydate
FROM
dbo.vGetVisits) Actions
WHERE
RowNum = 1
ORDER BY
mydate;
OPEN c ;
FETCH NEXT FROM c INTO @TheDate ;
WHILE @@FETCH_STATUS = 0
BEGIN
SET @Qualify = CASE WHEN DATEDIFF(day, @PreviousCheckDate, @Thedate) > 90 THEN 1 ELSE 0 END ;
IF @Qualify=1
BEGIN
INSERT @cd (TheDate, Qualify)
SELECT @TheDate, @Qualify ;
SET @PreviousCheckDate=@TheDate
END
FETCH NEXT FROM c INTO @TheDate ;
END
CLOSE c;
DEALLOCATE c;
SELECT TheDate
FROM @cd
ORDER BY TheDate ;
ইপারকিউবের DATEADD সমাধান ( ওয়াইপিসি, তারিখ )
DECLARE @cd TABLE
( TheDate datetime PRIMARY KEY,
Qualify INT NOT NULL
);
DECLARE
@TheDate DATETIME,
@Next_Date DATETIME,
@Interesting_Date DATETIME,
@Qualify INT = 0
DECLARE c CURSOR
LOCAL STATIC FORWARD_ONLY READ_ONLY
FOR
SELECT
[mydate]
FROM [test].[dbo].[vGetVisits]
ORDER BY mydate
;
OPEN c ;
FETCH NEXT FROM c INTO @TheDate ;
SET @Interesting_Date=@TheDate
INSERT @cd (TheDate, Qualify)
SELECT @TheDate, @Qualify ;
WHILE @@FETCH_STATUS = 0
BEGIN
IF @TheDate>DATEADD(DAY, 90, @Interesting_Date)
BEGIN
INSERT @cd (TheDate, Qualify)
SELECT @TheDate, @Qualify ;
SET @Interesting_Date=@TheDate;
END
FETCH NEXT FROM c INTO @TheDate;
END
CLOSE c;
DEALLOCATE c;
SELECT TheDate
FROM @cd
ORDER BY TheDate ;
পল হোয়াইটের সমাধান ( পিডাব্লু )
;WITH CTE AS
(
SELECT TOP (1)
T.[mydate]
FROM dbo.vGetVisits AS T
ORDER BY
T.[mydate]
UNION ALL
SELECT
SQ1.[mydate]
FROM
(
SELECT
T.[mydate],
rn = ROW_NUMBER() OVER (
ORDER BY T.[mydate])
FROM CTE
JOIN dbo.vGetVisits AS T
ON T.[mydate] > DATEADD(DAY, 90, CTE.[mydate])
) AS SQ1
WHERE
SQ1.rn = 1
)
SELECT
CTE.[mydate]
FROM CTE
OPTION (MAXRECURSION 0);
আমার তারিখের সমাধান ( পিএন, তারিখ )
DECLARE @cd TABLE
( TheDate datetime PRIMARY KEY
);
DECLARE @TheDate DATETIME
SET @TheDate=(SELECT MIN(mydate) as mydate FROM [dbo].[vGetVisits])
WHILE (@TheDate IS NOT NULL)
BEGIN
INSERT @cd (TheDate) SELECT @TheDate;
SET @TheDate=(
SELECT MIN(mydate) as mydate
FROM [dbo].[vGetVisits]
WHERE mydate>DATEADD(DAY, 90, @TheDate)
)
END
SELECT TheDate
FROM @cd
ORDER BY TheDate ;
আমার DATEDIFF সমাধান ( পিএন, DATEDIFF )
DECLARE @MinDate DATETIME;
SET @MinDate=(SELECT MIN(mydate) FROM dbo.vGetVisits);
;WITH gaps AS
(
SELECT
t1.[date]
, t1.[MaxDiff]
, SUM(t1.[MaxDiff]) OVER (ORDER BY t1.[date]) AS [Running Total]
FROM
(
SELECT
mydate AS [date]
, DATEDIFF(day,LAG(mydate, 1, mydate) OVER (ORDER BY mydate) , mydate) AS [MaxDiff]
FROM
(SELECT
RowNum = ROW_NUMBER() OVER(PARTITION BY cast(mydate as date) ORDER BY mydate)
, mydate
FROM dbo.vGetVisits
) Actions
WHERE RowNum = 1
) t1
)
SELECT [date]
FROM gaps t2
WHERE
( ([Running Total])%91 - ([Running Total]- [MaxDiff])%91 <0 )
OR
( [MaxDiff] > 90)
OR
([date]=@MinDate)
ORDER BY [date]
আমি এসকিউএল সার্ভার 2012 ব্যবহার করছি, তাই মিকেল এরিকসনের কাছে ক্ষমা চাইছি, তবে তার কোডটি এখানে পরীক্ষা করা হবে না। আমি এখনও DATADIFF এবং DATEADD এর সাথে তার সমাধানগুলি কিছু ডেটাসেটগুলিতে বিভিন্ন মান ফেরত পেতে আশা করব।
এবং আসল ফলাফলগুলি হ'ল: