IT

T-SQL을 사용하여 일 월과 연도의 날짜를 만듭니다.

lottoking 2020. 4. 2. 08:23
반응형

T-SQL을 사용하여 일 월과 연도의 날짜를 만듭니다.


2007 년 12 월 1 일과 같은 개별 부분이있는 날짜를 SQL Server 2005의 날짜 / 시간으로 변환하려고합니다. 다음을 시도했습니다.

CAST(DATEPART(year, DATE)+'-'+ DATEPART(month, DATE) +'-'+ DATEPART(day, DATE) AS DATETIME)

그러나 이것은 잘못된 날짜를 초래합니다. 세 날짜 값을 올바른 날짜 시간 형식으로 바꾸는 올바른 방법은 무엇입니까?


y, m, d모든 가정 int어떻습니까?

CAST(CAST(y AS varchar) + '-' + CAST(m AS varchar) + '-' + CAST(d AS varchar) AS DATETIME)

SQL Server 2012 이상에 대한 다른 답변참조하십시오


이 시도:

Declare @DayOfMonth TinyInt Set @DayOfMonth = 13
Declare @Month TinyInt Set @Month = 6
Declare @Year Integer Set @Year = 2006
-- ------------------------------------
Select DateAdd(day, @DayOfMonth - 1, 
          DateAdd(month, @Month - 1, 
              DateAdd(Year, @Year-1900, 0)))

잘 작동하고 문자열 변환을 수행하지 않는 이점이 추가되었으므로 순수한 산술 처리 (매우 빠름)이며 날짜 형식에 의존하지 않습니다. 이는 날짜 및 작은 날짜 시간 값에 대한 SQL Server의 내부 표현이 2라는 사실을 활용합니다. 부분 값 첫 번째 부분은 1900 년 1 월 1 일 이후의 일 수를 나타내는 정수이고, 두 번째 부분은 하루의 소수 부분을 나타내는 소수 부분입니다 (시간) --- 따라서 정수 값 0 (영 )는 항상 1900 년 1 월 1 일 자정 아침으로 직접 변환됩니다.

또는 @brinary의 제안 덕분에

Select DateAdd(yy, @Year-1900,  
       DateAdd(m,  @Month - 1, @DayOfMonth - 1)) 

@cade Roux에서 언급 한 것처럼 SQL 2012에는 이제
DATEFROMPARTS(year, month, day)
동일한 기능을 수행하는 기본 제공 기능이 있습니다.

@brinary가 제안한 마지막 솔루션 인 2016 년 10 월 3 일에 수정되었습니다. 년 추가를 먼저 수행하지 않으면 윤년 동안 작동하지 않는 것으로 보입니다.

select dateadd(month, @Month - 1, 
     dateadd(year, @Year-1900, @DayOfMonth - 1)); 

SQL Server 2012에는 훌륭하고 오랫동안 기다려온 새로운 DATEFROMPARTS 함수가 있습니다 (날짜가 유효하지 않으면 오류가 발생합니다-이 문제에 대한 DATEADD 기반 솔루션에 대한 나의 반대 의견).

http://msdn.microsoft.com/en-us/library/hh213228.aspx

DATEFROMPARTS(ycolumn, mcolumn, dcolumn)

또는

DATEFROMPARTS(@y, @m, @d)

또는 하나의 dateadd 함수 만 사용하십시오.

DECLARE @day int, @month int, @year int
SELECT @day = 4, @month = 3, @year = 2011

SELECT dateadd(mm, (@year - 1900) * 12 + @month - 1 , @day - 1)

Sql Server 2012에는 부품 ( DATEFROMPARTS )을 기반으로 날짜를 만드는 기능이 있습니다 . 우리의 나머지 부분을 위해, 파트에서 날짜를 결정하는 db 함수가 있습니다 (@Charles 덕분에) ...

IF EXISTS (SELECT * FROM dbo.sysobjects WHERE id = object_id(N'[dbo].[func_DateFromParts]'))
    DROP FUNCTION [dbo].[func_DateFromParts]
GO

CREATE FUNCTION [dbo].[func_DateFromParts]
(
    @Year INT,
    @Month INT,
    @DayOfMonth INT,
    @Hour INT = 0,  -- based on 24 hour clock (add 12 for PM :)
    @Min INT = 0,
    @Sec INT = 0
)
RETURNS DATETIME
AS
BEGIN

    RETURN DATEADD(second, @Sec, 
            DATEADD(minute, @Min, 
            DATEADD(hour, @Hour,
            DATEADD(day, @DayOfMonth - 1, 
            DATEADD(month, @Month - 1, 
            DATEADD(Year, @Year-1900, 0))))))

END

GO

당신은 이것을 이렇게 부를 수 있습니다 ...

SELECT dbo.func_DateFromParts(2013, 10, 4, 15, 50, DEFAULT)

보고...

2013-10-04 15:50:00.000

CAST 대신 CONVERT를 시도하십시오.

CONVERT는 날짜 형식을 나타내는 세 번째 매개 변수를 허용합니다.

형식 목록은 다음과 같습니다. http://msdn.microsoft.com/en-us/library/ms187928.aspx

다른 답변이 "올바른"답변으로 선택된 후 업데이트 :

이 제한을 나타내지 않고 서버의 NLS 설정에 따라 달라지는 응답이 선택된 이유를 실제로 이해하지 못합니다.


당신은 또한 사용할 수 있습니다

select DATEFROMPARTS(year, month, day) as ColDate, Col2, Col3 
From MyTable Where DATEFROMPARTS(year, month, day) Between @DateIni and @DateEnd

ver.2012 이후의 SQL 및 AzureSQL에서 작동


명시적인 시작점 '19000101'을 사용하는 것이 더 안전하고 깔끔합니다

create function dbo.fnDateTime2FromParts(@Year int, @Month int, @Day int, @Hour int, @Minute int, @Second int, @Nanosecond int)
returns datetime2
as
begin
    -- Note! SQL Server 2012 includes datetime2fromparts() function
    declare @output datetime2 = '19000101'
    set @output = dateadd(year      , @Year - 1900  , @output)
    set @output = dateadd(month     , @Month - 1    , @output)
    set @output = dateadd(day       , @Day - 1      , @output)
    set @output = dateadd(hour      , @Hour         , @output)
    set @output = dateadd(minute    , @Minute       , @output)
    set @output = dateadd(second    , @Second       , @output)
    set @output = dateadd(ns        , @Nanosecond   , @output)
    return @output
end

문자열을 유지하지 않으려면 다음과 같이 작동하십시오 (함수에 넣으십시오).

DECLARE @Day int, @Month int, @Year int
SELECT @Day = 1, @Month = 2, @Year = 2008

SELECT DateAdd(dd, @Day-1, DateAdd(mm, @Month -1, DateAdd(yy, @Year - 2000, '20000101')))

날짜 및 시간 부분 모두에서 날짜 시간이 필요한 경우 한 줄 솔루션을 추가합니다 .

select dateadd(month, (@Year -1900)*12 + @Month -1, @DayOfMonth -1) + dateadd(ss, @Hour*3600 + @Minute*60 + @Second, 0) + dateadd(ms, @Millisecond, 0)

시험

CAST(STR(DATEPART(year, DATE))+'-'+ STR(DATEPART(month, DATE)) +'-'+ STR(DATEPART(day, DATE)) AS DATETIME)

12 이하의 SQL Server 버전의 경우 다음 CAST과 함께 사용하는 것이 좋습니다.SET DATEFORMAT

-- 26 February 2015
SET DATEFORMAT dmy
SELECT CAST('26-2-2015' AS DATE)

SET DATEFORMAT ymd
SELECT CAST('2015-2-26' AS DATE)

그 문자열을 만드는 방법은 당신에게 달려 있습니다


이 쿼리를 시도하십시오 :

    SELECT SUBSTRING(CONVERT(VARCHAR,JOINGDATE,103),7,4)AS
    YEAR,SUBSTRING(CONVERT(VARCHAR,JOINGDATE,100),1,2)AS
MONTH,SUBSTRING(CONVERT(VARCHAR,JOINGDATE,100),4,3)AS DATE FROM EMPLOYEE1

결과:

2014    Ja    1
2015    Ja    1
2014    Ja    1
2015    Ja    1
2012    Ja    1
2010    Ja    1
2015    Ja    1

OP가 SQL 2005 답변을 요구한다는 것을 알고 있지만 질문은 꽤 오래되었으므로 SQL 2012 이상을 실행하는 경우 다음을 사용할 수 있습니다.

SELECT DATEADD(DAY, 1, EOMONTH(@somedate, -1))

참조 : https://docs.microsoft.com/en-us/sql/t-sql/functions/eomonth-transact-sql?view=sql-server-2017&viewFallbackFrom=sql-server-previousversions


개인적으로 하위 문자열을 선호하여 정리 옵션과 필요에 따라 문자열을 분할하는 기능을 제공합니다. 데이터는 'dd, mm, yyyy'형식이라고 가정합니다.

--2012 and above
SELECT CONCAT (
        RIGHT(REPLACE(@date, ' ', ''), 4)
        ,'-'
        ,RIGHT(CONCAT('00',SUBSTRING(REPLACE(@date, ' ', ''), CHARINDEX(',', REPLACE(@date, ' ', '')) + 1, LEN(REPLACE(@date, ' ', '')) - CHARINDEX(',', REPLACE(@date, ' ', '')) - 5)),2)
        ,'-'
        ,RIGHT(CONCAT('00',SUBSTRING(REPLACE(@date, ' ', ''), 1, CHARINDEX(',', REPLACE(@date, ' ', '')) - 1)),2)
        )

--2008 and below
SELECT   RIGHT(REPLACE(@date, ' ', ''), 4)
        +'-'
        +RIGHT('00'+SUBSTRING(REPLACE(@date, ' ', ''), CHARINDEX(',', REPLACE(@date, ' ', '')) + 1, LEN(REPLACE(@date, ' ', '')) - CHARINDEX(',', REPLACE(@date, ' ', '')) - 5),2)
        +'-'
        +RIGHT('00'+SUBSTRING(REPLACE(@date, ' ', ''), 1, CHARINDEX(',', REPLACE(@date, ' ', '')) - 1),2)

다음은 데이터가 열에 저장된 경우 어떻게 소송을 제기 할 수 있는지 보여줍니다. 말할 필요도없이, 컬럼에 적용하기 전에 결과 집합을 확인하는 것이 이상적입니다

DECLARE @Table TABLE (ID INT IDENTITY(1000,1), DateString VARCHAR(50), DateColumn DATE)

INSERT INTO @Table
SELECT'12, 1, 2007',NULL
UNION
SELECT'15,3, 2007',NULL
UNION
SELECT'18, 11 , 2007',NULL
UNION
SELECT'22 , 11, 2007',NULL
UNION
SELECT'30, 12, 2007  ',NULL

UPDATE @Table
SET DateColumn = CONCAT (
        RIGHT(REPLACE(DateString, ' ', ''), 4)
        ,'-'
        ,RIGHT(CONCAT('00',SUBSTRING(REPLACE(DateString, ' ', ''), CHARINDEX(',', REPLACE(DateString, ' ', '')) + 1, LEN(REPLACE(DateString, ' ', '')) - CHARINDEX(',', REPLACE(DateString, ' ', '')) - 5)),2)
        ,'-'
        ,RIGHT(CONCAT('00',SUBSTRING(REPLACE(DateString, ' ', ''), 1, CHARINDEX(',', REPLACE(DateString, ' ', '')) - 1)),2)
        ) 

SELECT ID,DateString,DateColumn
FROM @Table

참고 URL : https://stackoverflow.com/questions/266924/create-a-date-from-day-month-and-year-with-t-sql

반응형