Определить 3-ю пятницу каждого месяца

16

Мне нужно определить даты, которые являются «3-й пятницей каждого месяца» для диапазона дат «1.1.1996 - 30.8.2014» в SQL Server.

Я ожидаю, что я должен использовать комбинацию DENSE_RANK()и PARTITION BY()установить "rank = 3". Тем не менее, я новичок в SQL и не могу найти правильный код.

Emcor
источник

Ответы:

26

Данный:

  • Пятница называется "пятница"
  • Третья пятница месяца всегда выпадает с 15-го по 21-е число месяца.

    select thedate
    from yourtable
    where datename(weekday, thedate) = 'Friday'
    and datepart(day, thedate)>=15 and datepart(day, thedate)<=21;

Вы также можете использовать weekdayс datepart(), но это более читабельно с именем IMO. Сравнение строк, очевидно, будет медленнее.

Philᵀᴹ
источник
14

Чтобы получить независимый от языка / культуры ответ, необходимо учитывать разные названия дней недели и начала недели.

В Италии пятница - «Venerdì», а первый день недели - понедельник, а не воскресенье, как в США.

1900-01-01 Был понедельник, поэтому мы можем использовать эту информацию для расчета дня недели независимо от локали:

WITH dates AS (
    SELECT DATEADD(day, number, GETDATE()) AS theDate
    FROM master.dbo.spt_values
    WHERE type = 'P'
)
SELECT theDate, DATENAME(dw, theDate), DATEPART(dw, theDate)
FROM dates
WHERE DATEDIFF(day, '19000101', theDate) % 7 = 4
    AND DATEPART(day, thedate)>=15 and DATEPART(day, thedate)<=21;
spaghettidba
источник
12

Другой способ, который использует ответ Фила в качестве основы и позаботится о различных настройках:

select thedate
from yourtable
where (datepart(weekday, thedate) + @@DATEFIRST - 2) % 7 + 1 = 5   -- 5 -> Friday
  and (datepart(day, thedate) - 1) / 7 + 1 = 3 ;                   -- 3 -> 3rd week

5Код (если вы хотите , день недели , кроме пятницы) должен быть (так же , как SET DATEFIRSTкоды):

1 for Monday
2 for Tuesday
3 for Wednesday
4 for Thursday
5 for Friday
6 for Saturday
7 for Sunday

Вы также можете просто использовать "известную хорошую" дату, чтобы быть в безопасности перед лицом языковых настроек. Например, если вы ищете пятницу, проверьте календарь и убедитесь, что 2 января 2015 года было пятницей. Первое сравнение может быть записано как:

DATEPART(weekday,thedate) = DATEPART(weekday,'20150102') --Any Friday

Смотрите также Как получить N-й день недели от Питера Ларссона.

ypercubeᵀᴹ
источник
4

Я на самом деле написал статью об этом типе расчета здесь

В основном, вы можете использовать следующий код, чтобы найти 3-ю пятницу каждого месяца в любом диапазоне дат

USE TEMPDB
set nocount on;
IF OBJECT_ID('dbo.#t') is not null 
 DROP TABLE dbo.#t;
CREATE TABLE #t ([Date] datetime,
  [Year] smallint, [Quarter] tinyint, [Month] tinyint
, [Day] smallint -- from 1 to 366 = 1st to 366th day in a year
, [Week] tinyint -- from 1 to 54 = the 1st to 54th week in a year; 
, [Monthly_week] tinyint -- 1/2/3/4/5=1st/2nd/3rd/4th/5th week in a month
, [Week_day] tinyint -- 1=Mon, 2=Tue, 3=Wed, 4=Thu, 5=Fri, 6=Sat, 7=Sun
);
GO
USE TEMPDB
-- populate the table #t, and the day of week is defined as
-- 1=Mon, 2=Tue, 3=Wed, 4=Thu,5=Fri, 6=Sat, 7=Sun
;WITH   C0   AS (SELECT c FROM (VALUES(1),(1)) AS D(c)),
  C1   AS (SELECT 1 AS c FROM C0 AS A CROSS JOIN C0 AS B),
  C2   AS (SELECT 1 AS c FROM C1 AS A CROSS JOIN C1 AS B),
  C3   AS (SELECT 1 AS c FROM C2 AS A CROSS JOIN C2 AS B),
  C4   AS (SELECT 1 AS c FROM C3 AS A CROSS JOIN C3 AS B), 
  C5   AS (SELECT 1 AS c FROM C4 AS A CROSS JOIN C3 AS B),
  C6   AS (select rn=row_number() over (order by c)  from C5),
  C7   as (select [date]=dateadd(day, rn-1, '19000101') FROM C6 WHERE rn <= datediff(day, '19000101', '99991231')+1)

INSERT INTO #t ([year], [quarter], [month], [week], [day], [monthly_week], [week_day], [date])
SELECT datepart(yy, [DATE]), datepart(qq, [date]), datepart(mm, [date]), datepart(wk, [date])
     , datediff(day, dateadd(year, datediff(year, 0, [date]), 0), [date])+1
  , datepart(week, [date]) -datepart(week, dateadd(month, datediff(month, 0, [date]) , 0))+1
  , CASE WHEN datepart(dw, [date])+@@datefirst-1 > 7 THEN (datepart(dw, [date])+@@datefirst-1)%7
         ELSE datepart(dw, [date])+@@datefirst-1 END
 , [date]
FROM C7
    --where [date] between '19900101' and '20990101'; -- if you want to populate a range of dates
GO

select convert(char(10), [Date], 120) 
from #t
where Monthly_week=3
and week_day=5
and [date] between '2015-01-01' and '2015-12-31' -- change to your own date range
jyao
источник
2

Да, я знаю, что это старый пост. Думал, я бы по-другому взглянул на вещи, несмотря на ее возраст. Хех ... и мои извинения. Я только что понял, что почти скопировал то, что @jyao опубликовал выше.

Основываясь на текущем редактировании исходного вопроса ОП, я не мог понять, почему люди, публикующие ответы, делали это.

Просматривая правки, я нашел оригинальный вопрос и разместил его ниже ...

У меня есть временной ряд от 1.1.1996 до 30.8.2014 в базе данных SQL, например, с таблицей "db.dbo.datestable".

Мне нужно определить даты, которые являются «3-й пятницей каждого месяца» для этого диапазона дат в SQL.

Я ожидаю, что я должен использовать комбинацию "DENSE_RANK ()" и "PARTITION BY ()", чтобы установить "rank = 3". Тем не менее, я новичок в SQL и не могу найти правильный код.

Вы можете решить эту проблему?

Часть первоначального вопроса, которую я подчеркнул жирным шрифтом, кажется ключевой. Я, конечно, могу ошибаться, но мне кажется, что ОП заявлял, что у него есть таблица «Календарь» с именем «dbo.datestable», и для меня это имеет огромное значение, и теперь я понимаю, почему многие ответы это то, что они включают в себя то, что сгенерировало даты, потому что оно было опубликовано 10 ноября ... через день после окончательного редактирования вопроса, которое удалило последние следы ссылки на "dbo.datestable".

Как я уже сказал, я могу ошибаться, но вот моя интерпретация исходного вопроса.

У меня есть таблица «Календарь» под названием «dbo.datestable». Учитывая любой диапазон дат, охватываемых этой таблицей, как я могу вернуть только даты, которые являются 3-й пятницей каждого месяца в пределах данного диапазона дат?

Поскольку традиционные методы для этого уже были рассмотрены, я добавлю альтернативу, которая может быть полезна для некоторых.

Давайте смоделируем пару столбцов, которые, я думаю, ОП уже будут иметь в своей таблице. Конечно, я угадываю названия столбцов. Пожалуйста, добавьте все эквивалентные столбцы для вашей таблицы "Календарь". Кроме того, я делаю все это в TempDB, поэтому я не рискую вмешиваться в чью-то настоящую таблицу «Календарь».

--=================================================================================================
--      Simulate just a couple of the necessary columns of the OPs "Calendar" table.
--      This is not a part of the solution.  We're just trying to simulate what the OP has.
--=================================================================================================
--===== Variables to control the dates that will appear in the "Calendar" table.
DECLARE  @StartDT   DATETIME
        ,@EndDT     DATETIME
;
 SELECT  @StartDT = '1900' --Will be inclusive start of this year in calculations.
        ,@EndDT   = '2100' --Will be exclusive start of this year in calculations.
;
--===== Create the "Calendar" table with just enough columns to simulate the OP's.
 CREATE TABLE #datestable
        (
         TheDate    DATETIME NOT NULL
        ,DW         TINYINT  NOT NULL  --SQL standard abbreviate of "Day of Week"
        )
;
--===== Populate the "Calendar" table (uses "Minimal Logging" in 2008+ this case).    
   WITH cteGenDates AS
(
 SELECT TOP (DATEDIFF(dd,@StartDT,@EndDT)) --You can use "DAY" instead of "dd" if you prefer. I don't like it, though.
        TheDate = DATEADD(dd, ROW_NUMBER() OVER (ORDER BY (SELECT NULL))-1, @StartDT)
   FROM      sys.all_columns ac1
  CROSS JOIN sys.all_columns ac2
)
 INSERT INTO #datestable WITH (TABLOCK)
 SELECT  TheDate
        ,DW = DATEDIFF(dd,0,TheDate)%7+1 --Monday is 1, Friday is 5, Sunday is 7 etc.
   FROM cteGenDates
 OPTION (RECOMPILE) -- Help keep "Minimal Logging" in the presence of variables.
;
--===== Add the expected named PK for this example.
  ALTER TABLE #datestable 
    ADD CONSTRAINT PK_datestable PRIMARY KEY CLUSTERED (TheDate)
;

Я также не знаю, может ли ОП вносить изменения в свою таблицу «Календарь», так что это может не помочь ему, но может помочь другим. Имея это в виду, давайте добавим столбец DWoM (день недели для месяца). Если вам не нравится название, пожалуйста, не стесняйтесь сменить его на то, что вам нужно, на свой ящик.

--===== Add the new column.
  ALTER TABLE #datestable
    ADD DWOM TINYINT NOT NULL DEFAULT (0)
;

Далее нам нужно заполнить новый столбец. У ОП был смысл этого в его оригинальном посте.

--===== Populate the new column using the CTE trick for updates so that
     -- we can use a Windowing Function in an UPDATE.
   WITH cteGenDWOM AS
(
 SELECT DW# = ROW_NUMBER() OVER (PARTITION BY DATEDIFF(mm,0,TheDate), DW
                                     ORDER BY TheDate)
        ,DWOM
   FROM #datestable
)
 UPDATE cteGenDWOM
    SET DWOM = DW#
;

Теперь, поскольку это столбец фиксированной длины, который только что создал множество разделений страниц, поэтому нам нужно перестроить кластерный индекс, чтобы «перепаковать» таблицу, чтобы иметь как можно больше строк на страницу для повышения производительности.

--===== "Repack" the Clustered Index to get rid of the page splits we 
     -- caused by adding the new column.
  ALTER INDEX PK_datestable
     ON #datestable
        REBUILD WITH (FILLFACTOR = 100, SORT_IN_TEMPDB = ON)
;

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

--===== Return the 3rd Friday of every month included in the given date range.
 SELECT *
   FROM #datestable
  WHERE TheDate >= '1996-01-01' --I never use "BETWEEN" for dates out of habit for end date offsets.
    AND TheDate <= '2014-08-30'
    AND DW      =  5 --Friday
    AND DWOM    =  3 --The 3rd one for every month
  ORDER BY TheDate
;
Джефф Моден
источник
0

Вот простое решение для вырезания и вставки. Вы можете превратить это в функцию, если хотите.

Declare @CurrDate Date
Set @CurrDate = '11-20-2016'

declare @first datetime -- First of the month of interest (no time part)
declare @nth tinyint -- Which of them - 1st, 2nd, etc.
declare @dow tinyint -- Day of week we want
set @first = DATEFROMPARTS(YEAR(@CurrDate), MONTH(@CurrDate), 1) 
set @nth = 3
set @dow = 6
declare @result datetime
set @result = @first + 7*(@nth-1)
select  @result + (7 + @dow - datepart(weekday,@result))%7
jeffm
источник