Unix 타임 스탬프를 DateTime으로 또는 그 반대로 변환하려면 어떻게해야합니까?


754

이 예제 코드가 있지만 밀리 초 / 나노 초 문제에 대해 이야기하기 시작합니다.

C #의 Unix 시대 이후의 MSDN, Seconds 에도 같은 질문이 있습니다.

이것이 내가 지금까지 얻은 것입니다.

public Double CreatedEpoch
{
  get
  {
    DateTime epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0).ToLocalTime();
    TimeSpan span = (this.Created.ToLocalTime() - epoch);
    return span.TotalSeconds;
  }
  set
  {
    DateTime epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0).ToLocalTime();
    this.Created = epoch.AddSeconds(value);
  }
}

119
다가오는 .NET 4.6 (올해 후반에 출시 될 예정)에서는이를 지원합니다. DateTimeOffset.FromUnixTimeSecondsDateTimeOffset.ToUnixTimeSeconds방법을 참조하십시오 . 밀리 초 유닉스 타임 방법도 있습니다.
Jeppe Stig Nielsen

답변:


1017

필요한 것은 다음과 같습니다.

public static DateTime UnixTimeStampToDateTime( double unixTimeStamp )
{
    // Unix timestamp is seconds past epoch
    System.DateTime dtDateTime = new DateTime(1970,1,1,0,0,0,0,System.DateTimeKind.Utc);
    dtDateTime = dtDateTime.AddSeconds( unixTimeStamp ).ToLocalTime();
    return dtDateTime;
}

또는 Java의 경우 (타임 스탬프가 초가 아닌 밀리 초이기 때문에 다릅니다) :

public static DateTime JavaTimeStampToDateTime( double javaTimeStamp )
{
    // Java timestamp is milliseconds past epoch
    System.DateTime dtDateTime = new DateTime(1970,1,1,0,0,0,0,System.DateTimeKind.Utc);
    dtDateTime = dtDateTime.AddMilliseconds( javaTimeStamp ).ToLocalTime();
    return dtDateTime;
}

5
Windows의 시간은 HAL에 의해 처리되며 1ms ~ 15ms 내에서 정확합니다. 관심이 있으시면 Windows Internals 에서 더 많은 정보 를 참조하십시오.
Jim Schubert

16
이 답변은 초를 잘릴 위험이 있습니다. 이중은 부동 숫자입니다. 인수는 int / long / etc 여야합니다.
ccook

44
이러한 메소드는 double이 아닌 long 또는 int를 허용해야합니다. 또한 Java 타임 스탬프의 경우 1000으로 나눌 필요가 없습니다. 그냥 할dtDateTime.AddMilliseconds(javaTimeStamp).ToLocalTime();
저스틴 존슨

11
방금 "그 반대로"를 놓치셨습니까? DateTime을 타임 스탬프로 어떻게 변환합니까?
Jonny

38
닷넷 프레임 워크 4.6 이상 지금이 static DateTimeOffset.FromUnixMillisecondsDateTimeOffset.ToUnixMilliseconds.
rookie1024

421

.NET (V4.6)의 최신 버전을 추가했다 내장 된 유닉스 시간 변환을 지원합니다. 여기에는 초 또는 밀리 초로 표시되는 유닉스 시간과 출발 시간이 모두 포함됩니다.

  • UTC까지의 유닉스 시간 (초) DateTimeOffset:

DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeSeconds(1000);
  • DateTimeOffset 초 단위의 유닉스 시간 :

long unixTimeStampInSeconds = dateTimeOffset.ToUnixTimeSeconds();
  • 밀리 초 단위의 유닉스 시간 (UTC) DateTimeOffset:

DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeMilliseconds(1000000);
  • DateTimeOffset 밀리 초 단위의 유닉스 시간 :

long unixTimeStampInMilliseconds = dateTimeOffset.ToUnixTimeMilliseconds();

참고 :이 메소드는 UTC와 변환합니다 DateTimeOffset. DateTime표현 을 얻으려면 단순히 DateTimeOffset.UtcDateTime또는 DateTimeOffset.LocalDateTime속성을 사용하십시오 .

DateTime dateTime = dateTimeOffset.UtcDateTime;


시간을 현지 시간으로 변환하지 않습니다. DateTimeOffset.FromUnixTimeSeconds ()를 사용하면 UTC가 표시됩니다.
Berend de Boer

4
@BerenddeBoer 원한다면 사용할 수 있습니다 ToLocalTime.
i3arnon

1
현재 시간을 알기 위해long unixMilliseconds = DateTimeOffset.Now.ToUnixTimeMilliseconds();
Dan Diplo

219

DateTime에서 UNIX 타임 스탬프까지 :

public static double DateTimeToUnixTimestamp(DateTime dateTime)
{
    return (TimeZoneInfo.ConvertTimeToUtc(dateTime) - 
           new DateTime(1970, 1, 1, 0, 0, 0, 0, System.DateTimeKind.Utc)).TotalSeconds;
}

47

에서 위키 백과 :

UTC는 계절의 변화에 ​​따라 변경되지 않지만 시간대 관할 구역에서 일광 절약 시간 (서머 타임)을 준수하면 현지 시간 또는 시민 시간이 변경 될 수 있습니다. 예를 들어, 미국 동부 해안의 현지 시간은 겨울에는 UTC보다 5 시간이지만 일광 절약 시간은 4 시간입니다.

그래서 이것은 내 코드입니다.

TimeSpan span = (DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0,DateTimeKind.Utc));
double unixTime = span.TotalSeconds;

2
그러나 이것은 두 배를 반환합니다. 하나로 오래 캐스팅해야한다고 생각합니까?
knocte

30

밀리 초보다 높은 정밀도가 필요한 경우주의하십시오!

.NET (v4.6) 메소드 (예 : FromUnixTimeMilliseconds )는이 정밀도를 제공하지 않습니다.

AddSecondsAddMilliseconds 도 마이크로 초를 두 배로 줄였습니다.

이 버전은 높은 정밀도를 가지고 있습니다.

유닉스-> 날짜 / 시간

public static DateTime UnixTimestampToDateTime(double unixTime)
{
    DateTime unixStart = new DateTime(1970, 1, 1, 0, 0, 0, 0, System.DateTimeKind.Utc);
    long unixTimeStampInTicks = (long) (unixTime * TimeSpan.TicksPerSecond);
    return new DateTime(unixStart.Ticks + unixTimeStampInTicks, System.DateTimeKind.Utc);
}

DateTime-> 유닉스

public static double DateTimeToUnixTimestamp(DateTime dateTime)
{
    DateTime unixStart = new DateTime(1970, 1, 1, 0, 0, 0, 0, System.DateTimeKind.Utc);
    long unixTimeStampInTicks = (dateTime.ToUniversalTime() - unixStart).Ticks;
    return (double) unixTimeStampInTicks / TimeSpan.TicksPerSecond;
}

2
이것이 정답입니다. 다른 사람들은 타임 스탬프에서 다시 변환 할 때 시간대가 잘못됩니다.
IamIC

DateTime-> Java의 경우 [code] 만 반환 (long) unixTimeStampInTicks / TimeSpan.TicksPerMilliSecond; [/ code]
최대

귀하 UnixTimestampToDateTime의에서 제공된 시간 unixTime은 여전히 ​​몇 초입니다.
Ngoc Pham

@NgocPham 네, 그래요
Felix Keil

14

IdentityModel.EpochTimeExtensions를 참조하십시오.

public static class EpochTimeExtensions
{
    /// <summary>
    /// Converts the given date value to epoch time.
    /// </summary>
    public static long ToEpochTime(this DateTime dateTime)
    {
        var date = dateTime.ToUniversalTime();
        var ticks = date.Ticks - new DateTime(1970, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc).Ticks;
        var ts = ticks / TimeSpan.TicksPerSecond;
        return ts;
    }

    /// <summary>
    /// Converts the given date value to epoch time.
    /// </summary>
    public static long ToEpochTime(this DateTimeOffset dateTime)
    {
        var date = dateTime.ToUniversalTime();
        var ticks = date.Ticks - new DateTimeOffset(1970, 1, 1, 0, 0, 0, TimeSpan.Zero).Ticks;
        var ts = ticks / TimeSpan.TicksPerSecond;
        return ts;
    }

    /// <summary>
    /// Converts the given epoch time to a <see cref="DateTime"/> with <see cref="DateTimeKind.Utc"/> kind.
    /// </summary>
    public static DateTime ToDateTimeFromEpoch(this long intDate)
    {
        var timeInTicks = intDate * TimeSpan.TicksPerSecond;
        return new DateTime(1970, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc).AddTicks(timeInTicks);
    }

    /// <summary>
    /// Converts the given epoch time to a UTC <see cref="DateTimeOffset"/>.
    /// </summary>
    public static DateTimeOffset ToDateTimeOffsetFromEpoch(this long intDate)
    {
        var timeInTicks = intDate * TimeSpan.TicksPerSecond;
        return new DateTimeOffset(1970, 1, 1, 0, 0, 0, TimeSpan.Zero).AddTicks(timeInTicks);
    }
}

이것은 좋지만 작은 변경을 제안합니다. "long"유형의 사용은 "Int32"또는 "int"로 변경해야합니다. "긴"은 상당한 정밀도가 있음을 의미하지만 그렇지 않습니다. Int32는 유닉스 타임 스탬프에서 기대할 수있는 것을 더 암시 할 것입니다
JamesHoux

3
나는 이것이 DateTime.TicksInt64 (긴)이기 때문에 추가 검사되지 않은 캐스트를 피하고 있다고 생각합니다 .
orad

10

ScottCher의 답변을 보충하기 위해 최근에 입력 데이터 세트에 UNIX 타임 스탬프가 임의로 혼합되는 초와 밀리 초가 모두있는 성가신 시나리오에서 나 자신을 발견했습니다. 다음 코드는 이것을 잘 처리하는 것 같습니다.

static readonly DateTime UnixEpoch = new DateTime(1970, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc);
static readonly double MaxUnixSeconds = (DateTime.MaxValue - UnixEpoch).TotalSeconds;

public static DateTime UnixTimeStampToDateTime(double unixTimeStamp)
{
   return unixTimeStamp > MaxUnixSeconds
      ? UnixEpoch.AddMilliseconds(unixTimeStamp)
      : UnixEpoch.AddSeconds(unixTimeStamp);
}

1
생성 된 DateTime은 컴퓨터의 현지 시간 (코드 덕분에 Chris)이므로 DateTimeKind 인수를 사용하지 않을 때는주의하십시오!
Sam Grondahl

1
1978 년 1 월 11 일 이전의 밀리 초로 표시되는 날짜의 유닉스 타임 스탬프에는 작동하지 않습니다. 253324800 (초)의 유닉스 날짜 스탬프는 정확한 날짜 11.1978 년 1978 년을 제공하는 반면, 밀리 초 표현 253324800000은 날짜 18.07.9997을 제공합니다. 이것은 데이터 세트에서 효과가 있었을 수도 있지만 일반적인 해결책은 아닙니다.
Øyvind

8

Unix 시간 변환은 .NET Framework 4.6의 새로운 기능입니다.

이제 날짜 및 시간 값을 .NET Framework 유형 및 Unix 시간으로보다 쉽게 ​​변환 할 수 있습니다. 예를 들어 JavaScript 클라이언트와 .NET 서버간에 시간 값을 변환 할 때 필요할 수 있습니다. 다음 API가 DateTimeOffset 구조 에 추가되었습니다 .

static DateTimeOffset FromUnixTimeSeconds(long seconds)
static DateTimeOffset FromUnixTimeMilliseconds(long milliseconds)
long DateTimeOffset.ToUnixTimeSeconds()
long DateTimeOffset.ToUnixTimeMilliseconds()

이것은 현지 시간을 제공하지 않으며 UTC를 얻습니다.
Berend de Boer

@BerenddeBoer 합리적인 기본값입니다. 그 후 원하는대로 사용자 정의 오프셋을 적용 할 수 있습니다.
Deilan

1
@BerenddeBoer 유닉스 시간이 무엇인지 오해합니다. 유닉스 시간은 1970 년 1 월 1 일 자정 (UTC) 이후 초입니다. 당신이 어디에 있든 상관없이 그 시대는 바뀌지 않습니다. 사람이 읽을 수있는 현지 시간 표시로 변환하는 것은이 보편적 인 표현과는 별개입니다.
Tanktalus

5

현지 시간 조정없이 1/1/1970과의 변환을 비교하여 정답을 찾았습니다.

DateTime date = new DateTime(2011, 4, 1, 12, 0, 0, 0);
DateTime epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0);
TimeSpan span = (date - epoch);
double unixTime =span.TotalSeconds;

4
var dt = DateTime.Now; 
var unixTime = ((DateTimeOffset)dt).ToUnixTimeSeconds();

// 1510396991

var dt = DateTimeOffset.FromUnixTimeSeconds(1510396991);

// [11.11.2017 10:43:11 +00 : 00]


4

.net 4.6에서 다음을 수행 할 수 있습니다.

var dateTime = DateTimeOffset.FromUnixTimeSeconds(unixDateTime).DateTime;

3
DateTime unixEpoch = DateTime.ParseExact("1970-01-01", "yyyy-MM-dd", System.Globalization.CultureInfo.InvariantCulture);
DateTime convertedTime = unixEpoch.AddMilliseconds(unixTimeInMillisconds);

물론 unixEpoch전역 정적을 만들 수 있으므로 프로젝트에 한 번만 나타나고 AddSecondsUNIX 시간이 초이면 사용할 수 있습니다 .

다른 길로 가려면 :

double unixTimeInMilliseconds = timeToConvert.Subtract(unixEpoch).TotalMilliseconds;

Int64로 자르거나 TotalSeconds필요에 따라 사용하십시오 .


3

가장 효과적인 확장 프로그램을 작성했습니다. 누군가 그것을 찾으면 ...

public static class DateTimeExtensions
{
    public static DateTime FromUnixTimeStampToDateTime(this string unixTimeStamp)
    {

        return DateTimeOffset.FromUnixTimeSeconds(long.Parse(unixTimeStamp)).UtcDateTime;
    }
}


2

유닉스 틱은 1 초이며 (잘 기억하면) .NET 틱은 100 나노 초입니다.

나노초에 문제가 발생한 경우 AddTick (10000000 * value)을 사용해보십시오.


3
유닉스는 획기적인 초입니다-1/1/70입니다.
ScottCher

1

나는 변환 할 필요 timeval 형 구조체 포함 (초, 마이크로 초) UNIX time로를 DateTime정밀도를 잃지 않고하고 난 그냥 내 추가 할 수 있습니다 생각 때문에 여기에 대한 답을 발견하지 않았습니다 :

DateTime _epochTime = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);
private DateTime UnixTimeToDateTime(Timeval unixTime)
{
    return _epochTime.AddTicks(
        unixTime.Seconds * TimeSpan.TicksPerSecond +
        unixTime.Microseconds * TimeSpan.TicksPerMillisecond/1000);
}

1

DateTimeOffset 을 사용할 수 있습니다 .

예를 들어. DateTime 객체가 있습니다

var dateTime=new DateTime();

유닉스 타임 스탬프로 변환하려면 다음과 같이 달성 할 수 있습니다

var unixTimeSeconds= new DateTimeOffset(dateTime).ToUnixTimeSeconds()

자세한 내용은 다음 링크를 방문하십시오. DateTimeOffset.ToUnixTimeSeconds 메서드


0
public static class UnixTime
    {
        private static readonly DateTime Epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0);

        public static DateTime UnixTimeToDateTime(double unixTimeStamp)
        {
            return Epoch.AddSeconds(unixTimeStamp).ToUniversalTime();
        }
    }

UnixTime.UnixTimeToDateTime (double datetime))을 호출 할 수 있습니다


-2

.NET 4.6 이상의 경우 :

public static class UnixDateTime
{
    public static DateTimeOffset FromUnixTimeSeconds(long seconds)
    {
        if (seconds < -62135596800L || seconds > 253402300799L)
            throw new ArgumentOutOfRangeException("seconds", seconds, "");

        return new DateTimeOffset(seconds * 10000000L + 621355968000000000L, TimeSpan.Zero);
    }

    public static DateTimeOffset FromUnixTimeMilliseconds(long milliseconds)
    {
        if (milliseconds < -62135596800000L || milliseconds > 253402300799999L)
            throw new ArgumentOutOfRangeException("milliseconds", milliseconds, "");

        return new DateTimeOffset(milliseconds * 10000L + 621355968000000000L, TimeSpan.Zero);
    }

    public static long ToUnixTimeSeconds(this DateTimeOffset utcDateTime)
    {
        return utcDateTime.Ticks / 10000000L - 62135596800L;
    }

    public static long ToUnixTimeMilliseconds(this DateTimeOffset utcDateTime)
    {
        return utcDateTime.Ticks / 10000L - 62135596800000L;
    }

    [Test]
    public void UnixSeconds()
    {
        DateTime utcNow = DateTime.UtcNow;
        DateTimeOffset utcNowOffset = new DateTimeOffset(utcNow);

        long unixTimestampInSeconds = utcNowOffset.ToUnixTimeSeconds();

        DateTimeOffset utcNowOffsetTest = UnixDateTime.FromUnixTimeSeconds(unixTimestampInSeconds);

        Assert.AreEqual(utcNowOffset.Year, utcNowOffsetTest.Year);
        Assert.AreEqual(utcNowOffset.Month, utcNowOffsetTest.Month);
        Assert.AreEqual(utcNowOffset.Date, utcNowOffsetTest.Date);
        Assert.AreEqual(utcNowOffset.Hour, utcNowOffsetTest.Hour);
        Assert.AreEqual(utcNowOffset.Minute, utcNowOffsetTest.Minute);
        Assert.AreEqual(utcNowOffset.Second, utcNowOffsetTest.Second);
    }

    [Test]
    public void UnixMilliseconds()
    {
        DateTime utcNow = DateTime.UtcNow;
        DateTimeOffset utcNowOffset = new DateTimeOffset(utcNow);

        long unixTimestampInMilliseconds = utcNowOffset.ToUnixTimeMilliseconds();

        DateTimeOffset utcNowOffsetTest = UnixDateTime.FromUnixTimeMilliseconds(unixTimestampInMilliseconds);

        Assert.AreEqual(utcNowOffset.Year, utcNowOffsetTest.Year);
        Assert.AreEqual(utcNowOffset.Month, utcNowOffsetTest.Month);
        Assert.AreEqual(utcNowOffset.Date, utcNowOffsetTest.Date);
        Assert.AreEqual(utcNowOffset.Hour, utcNowOffsetTest.Hour);
        Assert.AreEqual(utcNowOffset.Minute, utcNowOffsetTest.Minute);
        Assert.AreEqual(utcNowOffset.Second, utcNowOffsetTest.Second);
        Assert.AreEqual(utcNowOffset.Millisecond, utcNowOffsetTest.Millisecond);
    }
}

4
난 이해가 안 돼요. .NET 4.6에서 BCL에는 이미 이러한 방법이 있습니다 (예 : 위의 질문에 대한 내 의견 또는 다른 새로운 답변 (2015) 참조) 따라서 다시 작성해야 할 요점은 무엇입니까? 4.6 이전 버전에 대한 솔루션
Jeppe Stig Nielsen
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.