Unix 타임스탬프를 DateTime으로 또는 그 반대로 변환하려면 어떻게 해야 합니까?
-
05-07-2019 - |
문제
이 예제 코드가 있지만 밀리초/나노초 문제에 대해 이야기하기 시작합니다.
MSDN에도 같은 질문이 있습니다. C#의 Unix 시대 이후 경과 시간(초).
이것이 내가 지금까지 얻은 것입니다:
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);
}
}
해결책
필요한 것은 다음과 같습니다.
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;
}
다른 팁
그만큼 .NET의 최신 버전 (v4.6) UNIX 시간 변환에 대한 내장 지원을 추가했습니다. 여기에는 UNIX 시간과의 시간 모두가 포함됩니다. 초 또는 밀리 초로 표시됩니다.
- UTC에 초의 유닉스 시간
DateTimeOffset
:
DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeSeconds(1000);
DateTimeOffset
몇 초 만에 unix 시간으로 :
long unixTimeStampInSeconds = dateTimeOffset.ToUnixTimeSeconds();
- UTC에서 밀리 초의 유닉스 시간
DateTimeOffset
:
DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeMilliseconds(1000000);
DateTimeOffset
밀리 초 단위의 unix 시간 :
long unixTimeStampInMilliseconds = dateTimeOffset.ToUnixTimeMilliseconds();
참고 :이 방법은 UTC로오고 DateTimeOffset
. 얻기 위해 DateTime
표현은 단순히 사용합니다 DateTimeOffset.UtcDateTime
또는 DateTimeOffset.LocalDateTime
속성:
DateTime dateTime = dateTimeOffset.UtcDateTime;
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;
}
UTC는 계절의 변화에 따라 변하지 않지만 시간대 관할권이 일광 절약 시간 (여름 시간)을 관찰하면 현지 시간이나 시민 시간이 바뀔 수 있습니다.예를 들어, UTC는 겨울 동안 미국 동부 해안에서 현지 시간보다 5 시간 앞서 있지만 일광 절약이 4 시간 앞서 있습니다.
이것이 내 코드입니다.
TimeSpan span = (DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0,DateTimeKind.Utc));
double unixTime = span.TotalSeconds;
밀리 초보다 정밀도가 필요한 경우 조심하십시오!
.NET (v4.6) 방법 (예 : inxtimeMilliseconds에서)이 정밀도를 제공하지 마십시오.
추가 그리고 Addmilliseconds 또한 이중으로 마이크로 초를 잘라냅니다.
이 버전은 정밀도가 높습니다.
UNIX-> DATETIME
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;
}
보다 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);
}
}
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);
}
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()
현지 시간 조정과 함께 전환을 1970 년 1 월 1 일 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;
DateTime unixEpoch = DateTime.ParseExact("1970-01-01", "yyyy-MM-dd", System.Globalization.CultureInfo.InvariantCulture);
DateTime convertedTime = unixEpoch.AddMilliseconds(unixTimeInMillisconds);
물론, 만들 수 있습니다 unixEpoch
글로벌 정적이므로 프로젝트에 한 번만 나타나면 사용할 수 있습니다. AddSeconds
유닉스 시간이 몇 초인 경우.
다른 방법으로 가려면 :
double unixTimeInMilliseconds = timeToConvert.Subtract(unixEpoch).TotalMilliseconds;
int64로 절단 및/또는 사용 TotalSeconds
필요에 따라.
유닉스 진드기는 1 초 (잘 기억된다면), .NET 진드기는 100 나노초입니다.
나노초에 문제가 발생했다면 AddTick (100000000 * 값)을 사용해보십시오.
var dt = DateTime.Now;
var unixTime = ((DateTimeOffset)dt).ToUnixTimeSeconds();
// 1510396991
var dt = DateTimeOffset.FromUnixTimeSeconds(1510396991);
// [11.11.2017 10:43:11 +00:00]
.NET 4.6에서 다음을 수행 할 수 있습니다.
var dateTime = DateTimeOffset.FromUnixTimeSeconds(unixDateTime).DateTime;
나는 변환해야했다 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);
}
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)을 호출 할 수 있습니다.
.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);
}
}