以毫秒为单位获取DateTime.Now


231

如何精确地以毫秒为单位构造实际时间的时间戳?

我需要类似16.4.2013 9:48:00:123的东西。这可能吗?我有一个应用程序,其中我每秒采样10次值,并且需要在图形中显示它们。



请注意,许多答案使用hh了几个小时(标准时间)。如果是时间戳记(以及许多其他情况),则应将其与tt(am / pm)配对或用HH(军事时间)代替。
戈茨

Answers:


304

如何精确地以毫秒为单位构造实际时间的时间戳?

我怀疑你是指毫秒精度DateTime具有很高的精度,但是在精度方面相当粗糙。一般来说,您不能。通常,系统时钟(DateTime.Now从那里获取数据)的分辨率约为10-15毫秒。有关更多详细信息,请参见Eric Lippert的有关精度和准确性博客文章

如果您需要比这更精确的计时,则可能要考虑使用NTP客户端。

但是,目前尚不清楚您是否真的需要毫秒级的精度。如果您不关心确切的时间-您只想以正确的顺序显示样本,并具有“非常好的”准确性,那么系统时钟应该就可以了。我建议您使用DateTime.UtcNow而不是DateTime.Now避免在夏时制转换周围出现时区问题等。

如果您的问题实际上只是将a转换为DateTime毫秒精度的字符串,我建议使用:

string timestamp = DateTime.UtcNow.ToString("yyyy-MM-dd HH:mm:ss.fff",
                                            CultureInfo.InvariantCulture);

(请注意,与您的样本不同,这是可排序的,并且不太可能引起有关是“月/日/年”还是“日/月/年”的混淆。)


5
@antonio:除了“不起作用”之外,我没有更多的信息可以提供。
乔恩·斯基特

是的,我可以提供更多信息:“ {0:yyyyMMdd HH:mm:ss.fff}”,DateTime.UtcNow-> 20180502 11:07:20.000 Win32.GetSystemTime(ref stime)-> 2018,2,5,11 ,7,20,0(ymdh mm ss ms)我可以创建一个以毫秒为单位的日期时间实例:var dt = new DateTime(2018,5,2,19,34,55,200); “ {0:yyyyMMdd HH:mm:ss.fff}”,dt-> 20180502 19:34:55.200 .Net紧凑框架,Windows Embedded Compact 7,在ARM Cortex-A8上运行
antonio,

@antonio:听起来您正在使用的.NET版本在所使用的硬件上根本无法提供亚秒级的精度-或您恰好在第二个边界上调用它。(如果您不断DateTime.UtcNow重复击打,它每秒只会增加一次吗?您没有显示出来。)
Jon Skeet

当我从DateTime.Now检索Ticks时,该值被截断:Ticks 636534596580000000 Date从Ticks 20180205 20:34:18.000。我认为限制是在硬件/软件中。我已经使用了Stopwatch类,而不是谢谢Jon
antonio '18

@antonio:这时可能值得提出一个新问题,其中包含所涉及设备的完整详细信息。
乔恩·斯基特

117

这应该工作:

DateTime.Now.ToString("hh.mm.ss.ffffff");

如果您不需要显示它并且只需要了解时差,那么不要将其转换为字符串。就这样吧DateTime.Now();

并用于TimeSpan了解时间间隔之间的差异:

DateTime start;
TimeSpan time;

start = DateTime.Now;

//Do something here

time = DateTime.Now - start;
label1.Text = String.Format("{0}.{1}", time.Seconds, time.Milliseconds.ToString().PadLeft(3, '0'));

我需要使用它进行计算,因此无法将其转换为字符串。
LuckyHK

25

根据该线程的建议,我一直在寻找类似的解决方案,我使用以下代码 DateTime.Now.ToString("MM/dd/yyyy hh:mm:ss.fff") ,它的工作方式就像魅力。注意:这.fff是您希望捕获的精度数字。


1
如果日期是下午(下午4点),则显示为早上。FIX:使用HH(军事时间)或添加tt(PM或AM)。如以下示例所示:repl.it/KGr3/1
Jaider

22

以毫秒为单位使用DateTime结构,格式如下:

string timestamp = DateTime.UtcNow.ToString("yyyy-MM-dd HH:mm:ss.fff", 
CultureInfo.InvariantCulture);
timestamp = timestamp.Replace("-", ".");

16

火烈鸟的答案对我来说似乎很好,但也许您想要:

DateTime.Now.Millisecond

但是,如果您要比较日期,则可以使用TimeSpan。


15
注意:如果您要比较两个DateTimes之间的时差作为TimeSpan,则需要TotalMilliseconds。毫秒保存当前的毫秒计数器,该计数器永远不大于1000,而TotalMilliseconds保存自从纪元以来经过的总毫秒数。
Contango 2014年

4

如果您仍然希望使用日期而不是字符串(如其他答案),则只需添加此扩展方法。

public static DateTime ToMillisecondPrecision(this DateTime d) {
    return new DateTime(d.Year, d.Month, d.Day, d.Hour, d.Minute,
                        d.Second, d.Millisecond, d.Kind);
}

3

据我了解的问题,您可以寻求:

DateTime dateTime = DateTime.Now;
DateTime dateTimeInMilliseconds = dateTime.AddTicks(-1 * dateTime.Ticks % 10000); 

这将切断小于1毫秒的滴答声。


是的,但DateTime.Now的实际分辨率可能不超过16毫秒。
彼得·莫滕森

2

麻烦 DateTime.UtcNowDateTime.Now在于,根据在计算机上和操作系统,它可能只精确到毫秒10和15之间。但是,在Windows计算机上,可以通过使用低级函数GetSystemTimePreciseAsFileTime来获得微秒精度,请参见GetTimeStamp()下面的函数。

    [System.Security.SuppressUnmanagedCodeSecurity, System.Runtime.InteropServices.DllImport("kernel32.dll")]
    static extern void GetSystemTimePreciseAsFileTime(out FileTime pFileTime);

    [System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential)]
    public struct FileTime  {
        public const long FILETIME_TO_DATETIMETICKS = 504911232000000000;   // 146097 = days in 400 year Gregorian calendar cycle. 504911232000000000 = 4 * 146097 * 86400 * 1E7
        public uint TimeLow;    // least significant digits
        public uint TimeHigh;   // most sifnificant digits
        public long TimeStamp_FileTimeTicks { get { return TimeHigh * 4294967296 + TimeLow; } }     // ticks since 1-Jan-1601 (1 tick = 100 nanosecs). 4294967296 = 2^32
        public DateTime dateTime { get { return new DateTime(TimeStamp_FileTimeTicks + FILETIME_TO_DATETIMETICKS); } }
    }

    public static DateTime GetTimeStamp() { 
        FileTime ft; GetSystemTimePreciseAsFileTime(out ft);
        return ft.dateTime;
    }

非常感谢。我已经搜索了很久了。它会引起性能问题吗?使用生产代码是否安全?
MuhammetGöktürkAyan


-2
public long millis() {
  return (long.MaxValue + DateTime.Now.ToBinary()) / 10000;
}

如果要微秒,只需更改1000010,如果要微秒的十分之一,请删除/ 10000


By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.