C#DateTime.Now精度

时间:2010-01-26 22:24:25

标签: c# .net datetime precision time-precision

我在进行一些单元测试时遇到了DateTime.UtcNow的一些意外行为。看起来,当您快速连续调用DateTime.Now/UtcNow时,它似乎会在超过预期的时间间隔内返回相同的值,而不是捕获更精确的毫秒增量。

我知道有一个秒表类更适合做精确的时间测量,但我很好奇是否有人可以在DateTime中解释这种行为?是否有针对DateTime.Now记录的官方精度(例如,精确到50毫秒内?)?为什么DateTime.Now会不像大多数CPU时钟那样精确?也许它只是为最低公分母CPU设计的?

public static void Main(string[] args)
{
    var stopwatch = new Stopwatch();
    stopwatch.Start();
    for (int i=0; i<1000; i++)
    {
        var now = DateTime.Now;
        Console.WriteLine(string.Format(
            "Ticks: {0}\tMilliseconds: {1}", now.Ticks, now.Millisecond));
    }

    stopwatch.Stop();
    Console.WriteLine("Stopwatch.ElapsedMilliseconds: {0}",
        stopwatch.ElapsedMilliseconds);

    Console.ReadLine();
}

7 个答案:

答案 0 :(得分:168)

  

为什么DateTime.Now会不像大多数CPU时钟那样精确?

一个好的时钟应该精确准确;那些是不同的。正如老笑话所说,一个停止的时钟每天准确两次,一分钟慢的时钟在任何时候都不准确。但是一分钟慢的时钟总是精确到最接近的分钟,而停止的时钟根本没有用的精确度。

为什么DateTime要精确,比如微秒,当它不可能准确到微秒?大多数人没有任何精确到微秒的官方时间信号源。因此,在 precision 的小数位后面给出六位数,其中最后五位是 garbage 撒谎

请记住,DateTime的目的是表示日期和时间。高精度计时并不完全是DateTime的目的;如你所知,这就是StopWatch的目的。 DateTime的目的是表示日期和时间,例如向用户显示当前时间,计算下周二之前的天数等等。

简而言之,“几点了?”并且“这需要多长时间?”是完全不同的问题;不要使用专门回答一个问题的工具来回答另一个问题。

感谢您的提问;这将是一篇很好的博客文章! : - )

答案 1 :(得分:18)

DateTime的精度在某种程度上与其运行的系统有关。精度与上下文切换的速度有关,该速度往往大约为15或16 ms。 (在我的系统上,我的测试实际上大约是14毫秒,但我看到一些笔记本电脑的精确度接近35-40毫秒。)

Peter Bromberg在C#中写了an article on high precision code timing,讨论了这个问题。

答案 2 :(得分:13)

我想要一个精确的Datetime.Now :),所以我把它煮熟了:

public class PreciseDatetime
{
    // using DateTime.Now resulted in many many log events with the same timestamp.
    // use static variables in case there are many instances of this class in use in the same program
    // (that way they will all be in sync)
    private static readonly Stopwatch myStopwatch = new Stopwatch();
    private static System.DateTime myStopwatchStartTime;

    static PreciseDatetime()
    {
        Reset();

        try
        {
            // In case the system clock gets updated
            SystemEvents.TimeChanged += SystemEvents_TimeChanged;
        }
        catch (Exception)
        {                
        }
    }

    static void SystemEvents_TimeChanged(object sender, EventArgs e)
    {
        Reset();
    }

    // SystemEvents.TimeChanged can be slow to fire (3 secs), so allow forcing of reset
    static public void Reset()
    {
        myStopwatchStartTime = System.DateTime.Now;
        myStopwatch.Restart();
    }

    public System.DateTime Now { get { return myStopwatchStartTime.Add(myStopwatch.Elapsed); } }
}

答案 3 :(得分:5)

由于缺乏实际检查.NET源代码,Eric Lippert对this SO question发表了评论,称DateTime仅精确到约30毫秒。用他的话说,不是纳秒准确的原因是它“不需要”。

答案 4 :(得分:5)

MSDN,您会发现DateTime.Now在所有NT操作系统上的近似分辨率为10毫秒。

实际精度取决于硬件。使用QueryPerformanceCounter可以获得更好的精度。

答案 5 :(得分:3)

From MSDN documentation

  

此属性的解析   取决于系统计时器。

他们还声称Windows NT 3.5及更高版本的近似分辨率为10毫秒:)

答案 6 :(得分:0)

  

此属性的分辨率取决于系统计时器,   取决于基础操作系统。它往往在0.5之间   和15毫秒。

结果,在较短的时间间隔(例如在循环中)重复调用Now属性可能会返回相同的值。

MSDN Link