UnixタイムスタンプをDateTimeに、またはその逆に変換するにはどうすればよいですか?


754

このサンプルコードがありますが、それからミリ秒/ナノ秒の問題について話し始めます。

同じ質問が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);
  }
}

119
次の.NET 4.6(今年後半にリリース予定)では、これに対するサポートが導入されています。メソッドDateTimeOffset.FromUnixTimeSecondsと参照してくださいDateTimeOffset.ToUnixTimeSeconds。ミリ秒のUNIX時間のメソッドもあります。
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によって処理され、1ミリ秒から15ミリ秒以内でほぼ正確です。詳細については、Windows Internalsの 112ページを参照してください。
ジム・シューベルト

16
この答えは秒を切り捨てるリスクがあります... doubleは浮動小数点数です。引数はint / long / etcでなければなりません。
2013年

44
これらのメソッドは、doubleではなくlongまたはintを受け入れる必要があります。また、Javaタイムスタンプの場合、1000で割り、四捨五入する必要はありません。Just dodtDateTime.AddMilliseconds(javaTimeStamp).ToLocalTime();
ジャスティンジョンソン、

11
「その逆」を逃しただけですか?DateTimeをタイムスタンプに変換するにはどうすればよいですか?
Jonny 14

38
.NET Framework 4.6以降の場合、現在static DateTimeOffset.FromUnixMillisecondsとがありDateTimeOffset.ToUnixMillisecondsます。
rookie1024 2016年

422

.NETの最新バージョン(V4.6)は、内蔵のUnixの時間の変換のためのサポートを追加しました。これには、秒単位またはミリ秒単位で表される、Unixの時刻の送受信が含まれます。

  • UTCまでの秒単位のUnix時間DateTimeOffset

DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeSeconds(1000);
  • DateTimeOffset 秒単位のUnix時間:

long unixTimeStampInSeconds = dateTimeOffset.ToUnixTimeSeconds();
  • UTCまでのミリ秒単位のUnix時間DateTimeOffset

DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeMilliseconds(1000000);
  • DateTimeOffset ミリ秒単位のUnix時間:

long unixTimeStampInMilliseconds = dateTimeOffset.ToUnixTimeMilliseconds();

注:これらのメソッドはUTCとの間で変換を行いDateTimeOffsetます。DateTime表現を取得するには、単にDateTimeOffset.UtcDateTimeまたはDateTimeOffset.LocalDateTimeプロパティを使用します。

DateTime dateTime = dateTimeOffset.UtcDateTime;


これは時間を現地時間に変換しません。DateTimeOffset.FromUnixTimeSeconds()を使用すると、UTCを取得します。
Berend de Boer 2017

4
@BerenddeBoer ToLocalTime必要に応じて使用できます。
i3arnon 2017

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 2013年

30

ミリ秒より高い精度が必要な場合は注意してください。

.NET(v4.6)メソッド(例:FromUnixTimeMilliseconds)は、この精度を提供しません。

AddSecondsAddMillisecondsも倍のマイクロ秒を切り捨てます。

これらのバージョンは高精度です。

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-> Unix

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 2017年

DateTime-> Javaの場合、[コード]を返すだけ(長い)unixTimeStampInTicks / TimeSpan.TicksPerMilliSecond; [/コード]
最大

のでUnixTimestampToDateTime、供給unixTimeはまだ数秒ですよね?
Ngoc Pham

@NgocPhamはい、そうです
フェリックス

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」に変更する必要があります。「長い」はかなりの精度があることを意味しますが、そうではありません。すべての計算は1秒までしか正確ではないため、Int32は、Unixタイムスタンプから予想されるものをより示唆するものになります
JamesHoux

3
DateTime.TicksInt64(long)が原因であると思うので、チェックされていない余分なキャストを回避しています。
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
DateTimeKind引数を使用しない場合は注意してください。構築されたDateTimeはコンピューターのローカル時間になるためです(コードのおかげでChris)。
Sam Grondahl 2013

1
注意してください-これがミリ秒で表されている場合、1978年1月11日より前の日付のUNIXタイムスタンプでは機能しません。UNIXの日付スタンプ253324800(秒)は11.01.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 2017

@BerenddeBoerこれは妥当なデフォルトです。その後、必要に応じてカスタムオフセットを適用できます。
Deilan 2017年

1
@BerenddeBoer UNIX時間とは何かを誤解しています。UNIX時間は、1970年1月1日、UTCの午前0時からの秒数です。あなたがどこにいるかは関係ありません、そのエポックからの秒数は変わりません。それを人間が読める現地時間表示に変換することは、この普遍的な表現とは異なります。
タンクタルス

5

変換を現地時間調整なしの1970年1月1日と比較するだけで、正しい答えが見つかりました。

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);

// [2017年11月11日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グローバルスタティックを作成できるので、プロジェクトに一度だけ表示する必要がありAddSeconds、UNIX時間が秒単位の場合に使用できます。

逆に行くには:

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

Unixの目盛りは1秒(よく覚えている場合)で、.NETの目盛りは100ナノ秒です。

ナノ秒の問題が発生した場合は、AddTick(10000000 *値)を使用してみてください。


3
Unixはエポックの数秒後です-これは1/1/70です。
ScottCher 2008年

1

精度を失うことなくを含むtimeval構造体(秒、マイクロ秒)を変換する必要があり、ここで答えを見つけられなかったので、私が追加するだけだと思いました:UNIX timeDateTime

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();

それをUnixタイムスタンプに変換したい場合は、次のようにして達成できます

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.