具有TimeSpan和自制计时器的计时器比秒表慢

时间:2015-02-08 19:29:37

标签: c# .net wpf

我正在尝试开发简单的计时器,它可以保存其最后一个值,并在新的应用程序启动时继续使用它。

Stopwatch类不可序列化,甚至无法初始化以便从特定时间开始。但它很棒。基准测试显示秒表的1分钟真的是1分钟。

我尝试以下列方式使用TimeSpan

private TimeSpan timerNew = new TimeSpan();
private DispatcherTimer dispatcherTimer = new DispatcherTimer();

public MainWindow()
{
    InitializeComponent();

    dispatcherTimer.Interval = TimeSpan.FromSeconds(1);
    dispatcherTimer.Tick += Timer_Tick;
}

private void Timer_Tick(object sender, EventArgs e)
{
    timerNew += new TimeSpan(0, 0, 0, 1);
    TbTimer.Text = String.Format("{0:00}:{1:00}:{2:00}",
        timerNew.Hours, timerNew.Minutes, timerNew.Seconds);
}

private void ButtonStart_OnClick(object sender, RoutedEventArgs e)
{
    dispatcherTimer.Start();
}

private void ButtonStop_OnClick(object sender, RoutedEventArgs e)
{
        dispatcherTimer.Stop();
}

private void ButtonReset_OnClick(object sender, RoutedEventArgs e)
{
    timerNew = new TimeSpan();
    TbTimer.Text = "00:00:00";
}

当我用真正的秒表检查时,我发现这个计时器实现每分钟丢失2秒。

我还尝试了自己的Timer实现,这是一个带有ulong字段的简单类,它在每个dispatcherTimer tick上递增。并且UI在将秒转换为小时,分钟等之后显示结果。但与真正的秒表相比,它每分钟也会损失2秒。

为什么这2秒钟丢失了?在可自定义的计时器中使用Stopwatch的替代方法是什么?

1 个答案:

答案 0 :(得分:3)

Windows线程调度程序不是"实时"调度程序,因为Windows不是"实时操作系统"。换句话说,所有的时间安排和调度都是在最好的努力下完成的。基础,没有任何精确的保证。此外,这总是会导致时间浪费,因为保证您的安排不会发生早期。因此,如果出现不精确的情况,那么它总是朝着"晚期"的方向发展。

Stopwatch类有效,因为它使用CPU支持的性能计数器,它不依赖于OS调度程序。硬件本身跟踪经过的时间并提供您需要的信息。

我建议反对使用DateTime.UtcNow来衡量已用时间,原因有二:首先,DateTime使用的时钟是可调的,因此即使使用UTC时间(至少可以补偿由于夏令时造成的自动调整,并不能保证准确。其次,您的特定方案似乎涉及一个问题,您希望序列化当前状态并将其恢复,DateTime.UtcNow无论如何都无法解决。

相反,您应该创建自己的可序列化秒表类,它使用Stopwatch本身作为基础,但它存储您添加到Stopwatch经过值的基本经过值

例如:

class SerializableStopwatch
{
    public TimeSpan BaseElapsed { get; set; }
    public TimeSpan Elapsed { get { return _stopwatch.Elapsed + BaseElapsed; } }

    private Stopwatch _stopwatch = new Stopwatch();

    // add whatever other members you want/need from the Stopwatch class,
    // simply delegating the operation to the _stopwatch member. For example:

    public void Start() { _stopwatch.Start(); }
    public void Stop() { _stopwatch.Stop(); }
    // etc.
}

您将如何序列化上述内容取决于您。在最简单的场景中,您可以将Elapsed属性格式化为字符串以保存值,然后当您想要还原对象时,解析该值,创建上述类的新实例,然后分配BaseElapsed属性的值。


有关该主题的其他讨论,您可能会发现Eric Lippert的博客文章Precision and accuracy of DateTime有用且有趣。