ฉันจะแปลงการประทับเวลา Unix เป็น DateTime และในทางกลับกันได้อย่างไร


754

มีรหัสตัวอย่างนี้ แต่จากนั้นจะเริ่มพูดถึงปัญหามิลลิวินาที / นาโนวินาที

คำถามเดียวกันคือใน MSDN, วินาทีตั้งแต่ยุค Unix ใน C #

นี่คือสิ่งที่ฉันได้รับ:

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วิธีการ มีวิธีการสำหรับมิลลิวินาทีเวลายูนิกซ์เช่นกัน
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รอบ ๆ หน้า 112 หากใครสนใจ
จิมชูเบิร์ต

16
คำตอบนี้มีความเสี่ยงในการตัดทอนวินาที ... คู่เป็นจำนวนลอยตัว อาร์กิวเมนต์ควรเป็น int / long / etc
ccook

44
วิธีการเหล่านี้ควรยอมรับความยาวหรือ int ไม่ใช่สองเท่า นอกจากนี้สำหรับการประทับเวลา Java ไม่จำเป็นต้องหารด้วย 1,000 และปัดเศษ เพิ่งทำdtDateTime.AddMilliseconds(javaTimeStamp).ToLocalTime();
จัสตินจอห์นสัน

11
คุณเพิ่งพลาด "ในทางกลับกัน"? เราจะแปลง DateTime เป็น timestamp อย่างไร
จอนนี่

38
สำหรับ .NET Framework 4.6 และสูงกว่าที่มีอยู่ในขณะนี้และstatic DateTimeOffset.FromUnixMilliseconds DateTimeOffset.ToUnixMilliseconds
rookie1024

421

รุ่นล่าสุดของ .NET (v4.6)ได้เพิ่มการสนับสนุนในตัวสำหรับการแปลงเวลายูนิกซ์ ซึ่งรวมถึงเวลาทั้งจากและเวลา Unix ที่แสดงด้วยวินาทีหรือมิลลิวินาที

  • เวลา Unix ในไม่กี่วินาทีถึง UTC DateTimeOffset:

DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeSeconds(1000);
  • DateTimeOffset ถึงเวลา Unix ในไม่กี่วินาที:

long unixTimeStampInSeconds = dateTimeOffset.ToUnixTimeSeconds();
  • เวลา Unix เป็นมิลลิวินาทีเป็น UTC DateTimeOffset:

DateTimeOffset dateTimeOffset = DateTimeOffset.FromUnixTimeMilliseconds(1000000);
  • DateTimeOffset ถึงเวลา Unix เป็นมิลลิวินาที:

long unixTimeStampInMilliseconds = dateTimeOffset.ToUnixTimeMilliseconds();

หมายเหตุ: วิธีการเหล่านี้แปลงไปและกลับจาก DateTimeOffsetUTC เราจะต้อง หากต้องการรับการDateTimeแสดงเพียงใช้DateTimeOffset.UtcDateTimeหรือDateTimeOffset.LocalDateTimeคุณสมบัติ:

DateTime dateTime = dateTimeOffset.UtcDateTime;


สิ่งนี้ไม่แปลงเวลาเป็นเวลาท้องถิ่น คุณได้รับ UTC หากคุณใช้ DateTimeOffset.FromUnixTimeSeconds ()
Berend de Boer

4
@BerenddeBoer คุณสามารถใช้ToLocalTimeถ้าคุณต้องการ
i3arnon

1
เพื่อให้ได้เวลาปัจจุบันคุณสามารถใช้long unixMilliseconds = DateTimeOffset.Now.ToUnixTimeMilliseconds();
Dan Diplo

219

DateTime ถึง UNIX Timestamp:

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

47

จากWikipedia :

UTC จะไม่เปลี่ยนแปลงเมื่อมีการเปลี่ยนแปลงฤดูกาล แต่เวลาท้องถิ่นหรือเวลาพลเมืองอาจเปลี่ยนแปลงได้หากเขตเวลาของเขตอำนาจสังเกตการปรับเวลาตามฤดูกาล (ฤดูร้อน) ตัวอย่างเช่นเวลาท้องถิ่นบนชายฝั่งตะวันออกของสหรัฐอเมริกาคือห้าชั่วโมงหลัง UTC ในช่วงฤดูหนาว แต่สี่ชั่วโมงหลังขณะที่มีการบันทึกแสงตามฤดูกาล

นี่คือรหัสของฉัน:

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 ) ไม่ได้ให้ความแม่นยำนี้

AddSecondsและ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 -> 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

สำหรับ DateTime-> Java เพียง [code] return (long) unixTimeStampInTicks / TimeSpan.TicksPerMilliSecond; [/ code]
สูงสุด

ดังนั้นในของคุณอุปกรณ์UnixTimestampToDateTimeที่ให้unixTimeยังคงอยู่ในไม่กี่วินาทีใช่ไหม
Ngoc Pham

@NgocPham ใช่มันเป็น
เฟลิกซ์ 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);
    }
}

สิ่งนี้ดี แต่ฉันขอแนะนำการเปลี่ยนแปลงเล็กน้อย: ควรใช้ประเภท "ยาว" เป็น "Int32" หรือ "int" "ยาว" แสดงถึงความแม่นยำที่สำคัญ แต่ไม่มี คณิตศาสตร์ทั้งหมดมีความแม่นยำเพียง 1 วินาทีดังนั้น Int32 จะแนะนำสิ่งที่คุณคาดหวังจากการประทับเวลา Unix
JamesHoux

3
ฉันคิดว่าเป็นเพราะDateTime.TicksInt64 (ยาว) ดังนั้นพวกเขาจึงหลีกเลี่ยงนักแสดงที่ไม่ได้ทำเครื่องหมายพิเศษ
orad

10

เพื่อเสริมคำตอบของ ScottCher เมื่อเร็ว ๆ นี้ฉันพบว่าตัวเองอยู่ในสถานการณ์ที่น่ารำคาญที่มีทั้งเวลาวินาทีและมิลลิวินาที UNIX timestamps ผสมกันโดยพลการในชุดข้อมูลอินพุต รหัสต่อไปนี้ดูเหมือนว่าจะจัดการได้ดี:

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

1
ระวัง - สิ่งนี้จะใช้ไม่ได้กับ unix timestamps สำหรับวันที่ก่อนวันที่ 11 มกราคม 1978 หากแสดงเป็นมิลลิวินาที unix datestamp ของ 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

@BerenddeBoer นั่นเป็นค่าเริ่มต้นที่สมเหตุสมผล คุณสามารถใช้การชดเชยแบบกำหนดเองหลังจากนั้นตามที่คุณต้องการ
Deilan

1
@BerenddeBoer นั่นเข้าใจผิดว่าเวลา unix คืออะไร เวลา Unix คือวินาทีตั้งแต่เที่ยงคืนวันที่ 1 มกราคม 1970 เวลา UTC ไม่สำคัญว่าคุณจะอยู่ที่ไหนจำนวนวินาทีตั้งแต่ยุคนั้นไม่เปลี่ยน การแปลงเป็นการแสดงเวลาท้องถิ่นที่มนุษย์สามารถอ่านได้นั้นแยกจากการเป็นตัวแทนสากลตามที่ควรจะเป็น
Tanktalus

5

ฉันพบคำตอบที่ถูกต้องโดยเปรียบเทียบการแปลงกับ 1/1/1970 w / o การปรับเวลาท้องถิ่น

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]



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 nanoseconds

หากคุณประสบปัญหากับนาโนวินาทีคุณอาจลองใช้ AddTick (ค่า 10,000000 *)


3
Unix คือวินาทีที่ผ่านมาในยุค - ซึ่งคือ 1/1/70
ScottCher

1

ผมจำเป็นต้องมีการแปลงstruct timeval (วินาที microseconds) ที่มี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 datetimeoffset

ตัวอย่างเช่น. ฉันมีวัตถุ DateTime

var dateTime=new DateTime();

หากฉันต้องการแปลงเป็น Unix time stamps ฉันสามารถทำได้ดังนี้

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.