Como posso obter o deslocamento correto entre o UTC e a hora local para uma data anterior ou posterior ao horário de verão?


29

Atualmente, uso o seguinte para obter um datetime local a partir de um UTC datetime:

SET @offset = DateDiff(minute, GetUTCDate(), GetDate())
SET @localDateTime = DateAdd(minute, @offset, @utcDateTime)

Meu problema é que, se o horário de verão ocorrer entre GetUTCDate()e @utcDateTime, @localDateTimetermina em uma hora de folga.

Existe uma maneira fácil de converter de utc para hora local em uma data que não seja a data atual?

Estou usando o SQL Server 2005

Respostas:


18

A melhor maneira de converter uma data UTC não atual em horário local é usar o CLR. O código em si é fácil; a parte difícil é convencer as pessoas de que o CLR não é puro mal ou assustador ...

Para um dos muitos exemplos, confira a postagem do blog de Harsh Chawla sobre o assunto .

Infelizmente, não há nada interno que possa lidar com esse tipo de conversão, exceto as soluções baseadas em CLR. Você poderia escrever uma função T-SQL que faça algo assim, mas teria que implementar a lógica de mudança de data por conta própria, e eu chamaria isso de decididamente não fácil.


Dada a complexidade real das variações regionais ao longo do tempo, dizer que "decididamente não é fácil" tentar isso em T-SQL puro provavelmente está subestimando ;-). Então, sim, o SQLCLR é o único meio confiável e eficiente de executar esta operação. +1 para isso. Para sua informação: a postagem do blog vinculado está funcionalmente correta, mas não segue as práticas recomendadas, portanto é infelizmente ineficiente. As funções de conversão entre o UTC e a hora local do servidor estão disponíveis na biblioteca SQL # (da qual sou autor), mas não na versão gratuita.
Solomon Rutzky

1
CLR fica mal quando deve ser adicionado WITH PERMISSION_SET = UNSAFE. Alguns ambientes não permitem isso como o AWS RDS. E é, bem, inseguro. Infelizmente, não existe uma implementação completa do fuso horário .Net que possa ser usada sem unsafepermissão. Veja aqui e aqui .
Frédéric

15

Eu desenvolvi e publiquei o projeto T-SQL Toolbox no codeplex para ajudar qualquer pessoa que lute com o processamento de data e hora e fuso horário no Microsoft SQL Server. É de código aberto e totalmente gratuito.

Oferece UDFs de conversão de data / hora fáceis usando T-SQL simples (sem CLRs), além de tabelas de configuração pré-preenchidas prontas para uso. E possui suporte completo ao horário de verão (DST).

Uma lista de todos os fusos horários suportados pode ser encontrada na tabela "DateTimeUtil.Timezone" (fornecida no banco de dados T-SQL Toolbox).

No seu exemplo, você pode usar o seguinte exemplo:

SELECT [DateTimeUtil].[UDF_ConvertUtcToLocalByTimezoneIdentifier] (
    'W. Europe Standard Time', -- the target local timezone
    '2014-03-30 00:55:00' -- the original UTC datetime you want to convert
)

Isso retornará o valor da data e hora local convertido.

Infelizmente, ele é suportado no SQL Server 2008 ou posterior apenas devido aos tipos de dados mais recentes (DATE, TIME, DATETIME2). Porém, como o código-fonte completo é fornecido, você pode ajustar facilmente as tabelas e UDFs, substituindo-os por DATETIME. Não tenho um MSSQL 2005 disponível para teste, mas ele deve funcionar com o MSSQL 2005 também. Em caso de perguntas, é só me avisar.


12

Eu sempre uso esse comando TSQL.

-- the utc value 
declare @utc datetime = '20/11/2014 05:14'

-- the local time

select DATEADD(hh, DATEDIFF(hh, getutcdate(), getdate()), @utc)

É muito simples e faz o trabalho.


2
Como há fusos horários que não são um deslocamento de hora inteiro do UTC, o uso deste DATEPART pode causar problemas.
Michael Green

4
Em relação ao comentário de Michael Green, você pode solucionar o problema alterando-o para SELECT DATEADD (MINUTE, DATEDIFF (MINUTE, GETUTCDATE (), GETDATE ()), @utc).
Usuário Registrado

4
Isso não funciona, pois você está apenas determinando se a hora atual é DST ou não e, em seguida, comparando uma hora que poderia ser DST ou não. Atualmente, o uso do código de exemplo acima e a data e hora no Reino Unido indicam que devem ser 6h14 da manhã; no entanto, novembro está fora do horário de verão; portanto, devem ser 5h14, conforme GMT e UTC coincidem.
Matt

Embora eu concorde que isso não atenda à pergunta real, no que diz respeito a esta resposta, acho que o seguinte é melhor: SELECT DATEADD (MINUTE, DATEPART (TZoffset, SYSDATETIMEOFFSET ()), @utc)
Eamon

@Ludo Bernaerts: primeiro uso em milissegundos, segundo: isso não funciona porque o deslocamento UTC hoje pode ser diferente do deslocamento UTC em um determinado horário (horário de verão - verão x inverno) ...
Quandary

11

Encontrei esta resposta no StackOverflow que fornece uma função definida pelo usuário que parece traduzir com precisão as datas e horários

A única coisa que você precisa modificar é a @offsetvariável na parte superior para defini-la como o deslocamento do fuso horário do servidor SQL executando esta função. No meu caso, nosso servidor SQL usa EST, que é GMT - 5

Não é perfeito e provavelmente não funcionará em muitos casos, com desvios de TZ de meia hora ou 15 minutos (para aqueles que eu recomendaria uma função CLR como Kevin recomendou ), mas funciona bem o suficiente para a maioria dos fusos horários genéricos no norte América.

CREATE FUNCTION [dbo].[UDTToLocalTime](@UDT AS DATETIME)  
RETURNS DATETIME
AS
BEGIN 
--====================================================
--Set the Timezone Offset (NOT During DST [Daylight Saving Time])
--====================================================
DECLARE @Offset AS SMALLINT
SET @Offset = -5

--====================================================
--Figure out the Offset Datetime
--====================================================
DECLARE @LocalDate AS DATETIME
SET @LocalDate = DATEADD(hh, @Offset, @UDT)

--====================================================
--Figure out the DST Offset for the UDT Datetime
--====================================================
DECLARE @DaylightSavingOffset AS SMALLINT
DECLARE @Year as SMALLINT
DECLARE @DSTStartDate AS DATETIME
DECLARE @DSTEndDate AS DATETIME
--Get Year
SET @Year = YEAR(@LocalDate)

--Get First Possible DST StartDay
IF (@Year > 2006) SET @DSTStartDate = CAST(@Year AS CHAR(4)) + '-03-08 02:00:00'
ELSE              SET @DSTStartDate = CAST(@Year AS CHAR(4)) + '-04-01 02:00:00'
--Get DST StartDate 
WHILE (DATENAME(dw, @DSTStartDate) <> 'sunday') SET @DSTStartDate = DATEADD(day, 1,@DSTStartDate)


--Get First Possible DST EndDate
IF (@Year > 2006) SET @DSTEndDate = CAST(@Year AS CHAR(4)) + '-11-01 02:00:00'
ELSE              SET @DSTEndDate = CAST(@Year AS CHAR(4)) + '-10-25 02:00:00'
--Get DST EndDate 
WHILE (DATENAME(dw, @DSTEndDate) <> 'sunday') SET @DSTEndDate = DATEADD(day,1,@DSTEndDate)

--Get DaylightSavingOffset
SET @DaylightSavingOffset = CASE WHEN @LocalDate BETWEEN @DSTStartDate AND @DSTEndDate THEN 1 ELSE 0 END

--====================================================
--Finally add the DST Offset 
--====================================================
RETURN DATEADD(hh, @DaylightSavingOffset, @LocalDate)
END



GO


3

Há algumas boas respostas para uma pergunta semelhante feita no Stack Overflow. Acabei usando uma abordagem T-SQL da segunda resposta de Bob Albright para limpar uma bagunça causada por um consultor de conversão de dados.

Ele funcionou para quase todos os nossos dados, mas depois percebi que seu algoritmo só funciona em datas já em 5 de abril de 1987 , e tínhamos algumas datas da década de 1940 que ainda não foram convertidas corretamente. Finalmente, precisávamos das UTCdatas em nosso banco de dados do SQL Server para alinhar com um algoritmo em um programa de terceiros que usava a API Java para converter UTCpara a hora local.

Gosto do CLRexemplo da resposta de Kevin Feasel acima, usando o exemplo de Harsh Chawla, e também gostaria de compará-lo a uma solução que usa Java, já que nosso front end usa Java para fazer a UTCconversão para a hora local.

A Wikipedia menciona oito emendas constitucionais diferentes que envolvem ajustes de fuso horário anteriores a 1987, e muitas delas são muito localizadas em estados diferentes; portanto, há uma chance de que o CLR e Java possam interpretá-las de maneira diferente. Seu código de aplicativo front-end usa dotnet ou Java ou as datas anteriores a 1987 são um problema para você?


2

Você pode fazer isso facilmente com um procedimento armazenado CLR.

[SqlFunction]
public static SqlDateTime ToLocalTime(SqlDateTime UtcTime, SqlString TimeZoneId)
{
    if (UtcTime.IsNull)
        return UtcTime;

    var timeZone = TimeZoneInfo.FindSystemTimeZoneById(TimeZoneId.Value);
    var localTime = TimeZoneInfo.ConvertTimeFromUtc(UtcTime.Value, timeZone);
    return new SqlDateTime(localTime);
}

Você pode armazenar os fusos horários disponíveis em uma tabela:

CREATE TABLE TimeZones
(
    TimeZoneId NVARCHAR(32) NOT NULL CONSTRAINT PK_TimeZones PRIMARY KEY,
    DisplayName NVARCHAR(64) NOT NULL,
    SupportsDaylightSavingTime BIT NOT NULL,
)

E esse procedimento armazenado preencherá a tabela com os possíveis fusos horários no seu servidor.

public partial class StoredProcedures
{
    [SqlProcedure]
    public static void PopulateTimezones()
    {
        using (var sql = new SqlConnection("Context Connection=True"))
        {
            sql.Open();

            using (var cmd = sql.CreateCommand())
            {
                cmd.CommandText = "DELETE FROM TimeZones";
                cmd.ExecuteNonQuery();

                cmd.CommandText = "INSERT INTO [dbo].[TimeZones]([TimeZoneId], [DisplayName], [SupportsDaylightSavingTime]) VALUES(@TimeZoneId, @DisplayName, @SupportsDaylightSavingTime);";
                var Id = cmd.Parameters.Add("@TimeZoneId", SqlDbType.NVarChar);
                var DisplayName = cmd.Parameters.Add("@DisplayName", SqlDbType.NVarChar);
                var SupportsDaylightSavingTime = cmd.Parameters.Add("@SupportsDaylightSavingTime", SqlDbType.Bit);

                foreach (var zone in TimeZoneInfo.GetSystemTimeZones())
                {
                    Id.Value = zone.Id;
                    DisplayName.Value = zone.DisplayName;
                    SupportsDaylightSavingTime.Value = zone.SupportsDaylightSavingTime;

                    cmd.ExecuteNonQuery();
                }
            }
        }
    }
}

CLR fica mal quando deve ser adicionado WITH PERMISSION_SET = UNSAFE. Alguns ambientes não permitem isso como o AWS RDS. E é, bem, inseguro. Infelizmente, não há uma implementação completa do fuso horário .Net que possa ser usada sem unsafepermissão. Veja aqui e aqui .
Frédéric

2

O SQL Server versão 2016 resolverá esse problema de uma vez por todas . Para versões anteriores, uma solução CLR é provavelmente mais fácil. Ou para uma regra específica de horário de verão (como apenas nos EUA), uma função T-SQL pode ser relativamente simples.

No entanto, acho que uma solução T-SQL genérica pode ser possível. Desde que xp_regreadfuncione, tente o seguinte:

CREATE TABLE #tztable (Value varchar(50), Data binary(56));
DECLARE @tzname varchar(150) = 'SYSTEM\CurrentControlSet\Control\TimeZoneInformation'
EXEC master.dbo.xp_regread 'HKEY_LOCAL_MACHINE', @tzname, 'TimeZoneKeyName', @tzname OUT;
SELECT @tzname = 'SOFTWARE\Microsoft\Windows NT\CurrentVersion\Time Zones\' + @tzname
INSERT INTO #tztable
EXEC master.dbo.xp_regread 'HKEY_LOCAL_MACHINE', @tzname, 'TZI';
SELECT                                                                                  -- See http://msdn.microsoft.com/ms725481
 CAST(CAST(REVERSE(SUBSTRING(Data,  1, 4)) AS binary(4))      AS int) AS BiasMinutes,   -- UTC = local + bias: > 0 in US, < 0 in Europe!
 CAST(CAST(REVERSE(SUBSTRING(Data,  5, 4)) AS binary(4))      AS int) AS ExtraBias_Std, --   0 for most timezones
 CAST(CAST(REVERSE(SUBSTRING(Data,  9, 4)) AS binary(4))      AS int) AS ExtraBias_DST, -- -60 for most timezones: DST makes UTC 1 hour earlier
 -- When DST ends:
 CAST(CAST(REVERSE(SUBSTRING(Data, 13, 2)) AS binary(2)) AS smallint) AS StdYear,       -- 0 = yearly (else once)
 CAST(CAST(REVERSE(SUBSTRING(Data, 15, 2)) AS binary(2)) AS smallint) AS StdMonth,      -- 0 = no DST
 CAST(CAST(REVERSE(SUBSTRING(Data, 17, 2)) AS binary(2)) AS smallint) AS StdDayOfWeek,  -- 0 = Sunday to 6 = Saturday
 CAST(CAST(REVERSE(SUBSTRING(Data, 19, 2)) AS binary(2)) AS smallint) AS StdWeek,       -- 1 to 4, or 5 = last <DayOfWeek> of <Month>
 CAST(CAST(REVERSE(SUBSTRING(Data, 21, 2)) AS binary(2)) AS smallint) AS StdHour,       -- Local time
 CAST(CAST(REVERSE(SUBSTRING(Data, 23, 2)) AS binary(2)) AS smallint) AS StdMinute,
 CAST(CAST(REVERSE(SUBSTRING(Data, 25, 2)) AS binary(2)) AS smallint) AS StdSecond,
 CAST(CAST(REVERSE(SUBSTRING(Data, 27, 2)) AS binary(2)) AS smallint) AS StdMillisec,
 -- When DST starts:
 CAST(CAST(REVERSE(SUBSTRING(Data, 29, 2)) AS binary(2)) AS smallint) AS DSTYear,       -- See above
 CAST(CAST(REVERSE(SUBSTRING(Data, 31, 2)) AS binary(2)) AS smallint) AS DSTMonth,
 CAST(CAST(REVERSE(SUBSTRING(Data, 33, 2)) AS binary(2)) AS smallint) AS DSTDayOfWeek,
 CAST(CAST(REVERSE(SUBSTRING(Data, 35, 2)) AS binary(2)) AS smallint) AS DSTWeek,
 CAST(CAST(REVERSE(SUBSTRING(Data, 37, 2)) AS binary(2)) AS smallint) AS DSTHour,
 CAST(CAST(REVERSE(SUBSTRING(Data, 39, 2)) AS binary(2)) AS smallint) AS DSTMinute,
 CAST(CAST(REVERSE(SUBSTRING(Data, 41, 2)) AS binary(2)) AS smallint) AS DSTSecond,
 CAST(CAST(REVERSE(SUBSTRING(Data, 43, 2)) AS binary(2)) AS smallint) AS DSTMillisec
FROM #tztable;
DROP TABLE #tztable

Uma função T-SQL (complexa) pode usar esses dados para determinar o deslocamento exato para todas as datas durante a regra de horário de verão atual.


2
DECLARE @TimeZone VARCHAR(50)
EXEC MASTER.dbo.xp_regread 'HKEY_LOCAL_MACHINE', 'SYSTEM\CurrentControlSet\Control\TimeZoneInformation', 'TimeZoneKeyName', @TimeZone OUT
SELECT @TimeZone
DECLARE @someUtcTime DATETIME
SET @someUtcTime = '2017-03-05 15:15:15'
DECLARE @TimeBiasAtSomeUtcTime INT
SELECT @TimeBiasAtSomeUtcTime = DATEDIFF(MINUTE, @someUtcTime, @someUtcTime AT TIME ZONE @TimeZone)
SELECT DATEADD(MINUTE, @TimeBiasAtSomeUtcTime * -1, @someUtcTime)

2
Oi Joost! Obrigado por publicar. Se você adicionar alguma explicação à sua resposta, pode ser muito mais fácil de entender.
LowlyDBA

2

Aqui está uma resposta escrita para um aplicativo específico do Reino Unido e baseada exclusivamente no SELECT.

  1. Sem deslocamento de fuso horário (por exemplo, Reino Unido)
  2. Escrito para horário de verão, começando no último domingo de março e terminando no último domingo de outubro (regras do Reino Unido)
  3. Não aplicável entre meia-noite e 01:00 no horário de verão. Isso pode ser corrigido, mas o aplicativo para o qual foi escrito não exige.

    -- A variable holding an example UTC datetime in the UK, try some different values:
    DECLARE
    @App_Date datetime;
    set @App_Date = '20250704 09:00:00'
    
    -- Outputting the local datetime in the UK, allowing for daylight saving:
    SELECT
    case
    when @App_Date >= dateadd(day, 1 - datepart(weekday, dateadd(day, -1, dateadd(month, 3, dateadd(year, datediff(year, 0, @App_Date), 0)))), dateadd(day, -1, dateadd(month, 3, dateadd(year, datediff(year, 0, @App_Date), 0))))
        and @App_Date < dateadd(day, 1 - datepart(weekday, dateadd(day, -1, dateadd(month, 10, dateadd(year, datediff(year, 0, @App_Date), 0)))), dateadd(day, -1, dateadd(month, 10, dateadd(year, datediff(year, 0, @App_Date), 0))))
        then DATEADD(hour, 1, @App_Date) 
    else @App_Date 
    end

Você pode considerar usar os nomes das peças com data longa, em vez dos nomes curtos. Apenas para maior clareza. Ver o excelente artigo de Aaron Bertrand em diversos "maus hábitos"
Max Vernon

Além disso, seja bem-vindo aos administradores de banco de dados - faça o tour, se ainda não o fez!
Max Vernon

1
Obrigado a todos, comentários úteis e sugestões úteis de edição, sou um novato aqui, de alguma forma eu consegui acumular 1 ponto que é fabuloso :-).
Colinp_1 # 10/17

agora você tem 11.
Max Vernon
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.