- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
首先,让我说一下,我知道 Windows 不是实时操作系统。我不希望任何给定的定时器间隔精确到 15 毫秒。然而,对我来说重要的是我的计时器滴答总和从长远来看是准确的,这对于股票计时器或秒表来说是不可能的。
如果将普通的 System.Timers.Timer(或者,更糟糕的是,Windows.Forms.Timer)配置为每 100 毫秒运行一次,它会在 100 毫秒到大约 115 毫秒之间的某个时间滴答作响。这意味着如果我需要做一天的事件,10 * 60 * 60 * 24 = 864,000 次,我可能会迟到三个多小时!这对我的申请来说是 Not Acceptable 。
我怎样才能最好地构建一个平均持续时间准确的计时器?我已经构建了一个,但我认为它可以更流畅、更有效,或者……必须有更好的方法。为什么 .NET 会提供低精度 DateTime.Now、低精度 Timer 和高精度 Diagnostics.Stopwatch,但没有高精度计时器?
我写了下面的定时器更新例程:
var CorrectiveStopwatch = new System.Diagnostics.Stopwatch();
var CorrectedTimer = new System.Timers.Timer()
{
Interval = targetInterval,
AutoReset = true,
};
CorrectedTimer.Elapsed += (o, e) =>
{
var actualMilliseconds = ;
// Adjust the next tick so that it's accurate
// EG: Stopwatch says we're at 2015 ms, we should be at 2000 ms
// 2000 + 100 - 2015 = 85 and should trigger at the right time
var newInterval = StopwatchCorrectedElapsedMilliseconds +
targetInterval -
CorrectiveStopwatch.ElapsedMilliseconds;
// If we're over 1 target interval too slow, trigger ASAP!
if (newInterval <= 0)
{
newInterval = 1;
}
CorrectedTimer.Interval = newInterval;
StopwatchCorrectedElapsedMilliseconds += targetInterval;
};
如您所见,它不太可能非常顺利。而且,在测试中,它一点也不流畅!我在一个示例运行中得到以下值:
100
87
91
103
88
94
102
93
103
88
此序列的平均值为 我了解网络时间和其他工具会平滑地强制当前时间以获得准确的计时器。也许执行某种 PID 循环以使目标间隔达到 targetInterval - 15/2 左右可能需要更小的更新,或者平滑可能会减少短事件和长事件之间的时间差异。我如何将其与我的方法相结合?是否有现有的图书馆可以做到这一点?
我将它与普通秒表进行了比较,它似乎是准确的。 5 分钟后,我测量了以下毫秒计数器值:
DateTime elapsed: 300119
Stopwatch elapsed: 300131
Timer sum: 274800
Corrected sum: 300200
30 分钟后,我测量了:
DateTime elapsed: 1800208.2664
Stopwatch elapsed: 1800217
Timer sum: 1648500
Corrected sum: 1800300
股票计时器比秒表慢 151.8 秒,占总数的 8.4%。这似乎非常接近 100 到 115 毫秒之间的平均滴答持续时间!此外,校正后的总和仍在测量值的 100 毫秒以内(这些比我用 watch 发现的误差更准确),所以这是可行的。
我的完整测试程序如下。它在 Windows 窗体中运行,需要“Form1”上的文本框,或者删除 textBox1.Text =
行并在命令行项目中运行它。
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace StopwatchTimerTest
{
public partial class Form1 : Form
{
private System.Diagnostics.Stopwatch ElapsedTimeStopwatch;
private double TimerSumElapsedMilliseconds;
private double StopwatchCorrectedElapsedMilliseconds;
private DateTime StartTime;
public Form1()
{
InitializeComponent();
Run();
}
private void Run()
{
var targetInterval = 100;
// A stopwatch running in normal mode provides the correct elapsed time to the best abilities of the system
// (save, for example, a network time server or GPS peripheral)
ElapsedTimeStopwatch = new System.Diagnostics.Stopwatch();
// A normal System.Timers.Timer ticks somewhere between 100 ms and 115ms, quickly becoming inaccurate.
var NormalTimer = new System.Timers.Timer()
{
Interval = targetInterval,
AutoReset = true,
};
NormalTimer.Elapsed += (o, e) =>
{
TimerSumElapsedMilliseconds += NormalTimer.Interval;
};
// This is my "quick and dirty" corrective stopwatch
var CorrectiveStopwatch = new System.Diagnostics.Stopwatch();
var CorrectedTimer = new System.Timers.Timer()
{
Interval = targetInterval,
AutoReset = true,
};
CorrectedTimer.Elapsed += (o, e) =>
{
var actualMilliseconds = CorrectiveStopwatch.ElapsedMilliseconds; // Drop this into a variable to avoid confusion in the debugger
// Adjust the next tick so that it's accurate
// EG: Stopwatch says we're at 2015 ms, we should be at 2000 ms, 2000 + 100 - 2015 = 85 and should trigger at the right time
var newInterval = StopwatchCorrectedElapsedMilliseconds + targetInterval - actualMilliseconds;
if (newInterval <= 0)
{
newInterval = 1; // Basically trigger instantly in an attempt to catch up; could be smoother...
}
CorrectedTimer.Interval = newInterval;
// Note: could be more accurate with actual interval, but actual app uses sum of tick events for many things
StopwatchCorrectedElapsedMilliseconds += targetInterval;
};
var updateTimer = new System.Windows.Forms.Timer()
{
Interval = 1000,
Enabled = true,
};
updateTimer.Tick += (o, e) =>
{
var result = "DateTime elapsed: " + (DateTime.Now - StartTime).TotalMilliseconds.ToString() + Environment.NewLine +
"Stopwatch elapsed: " + ElapsedTimeStopwatch.ElapsedMilliseconds.ToString() + Environment.NewLine +
"Timer sum: " + TimerSumElapsedMilliseconds.ToString() + Environment.NewLine +
"Corrected sum: " + StopwatchCorrectedElapsedMilliseconds.ToString();
Console.WriteLine(result + Environment.NewLine);
textBox1.Text = result;
};
// Start everything simultaneously
StartTime = DateTime.Now;
ElapsedTimeStopwatch.Start();
updateTimer.Start();
NormalTimer.Start();
CorrectedTimer.Start();
CorrectiveStopwatch.Start();
}
}
}
最佳答案
Windows is not a real-time OS
更准确地说,在保护模式请求分页虚拟内存操作系统的第 3 环中运行的代码不能假定它可以提供任何执行保证。像页面错误这样简单的事情会大大延迟代码执行。更像是这样,第 2 代垃圾收集总是会破坏您的期望。 Ring 0 代码(驱动程序)通常具有出色的中断响应时间,模数一个在其基准测试中作弊的蹩脚视频驱动程序。
所以不,希望由 Timer 激活的代码将立即运行通常是空想。总会有延迟。 DispatcherTimer 或 Winforms 计时器只能在 UI 线程空闲时计时,异步计时器需要与所有其他线程池请求以及竞争处理器的其他进程拥有的线程竞争。
因此,您永远不想做的一件事是在事件处理程序或回调中增加耗时计数器。它会很快偏离流逝的挂钟时间。以一种完全不可预测的方式,机器负载越重,它就越落后。
解决方法很简单,如果您关心实际耗时,则使用时钟。无论是 DateTime.UtcNow 还是 Environment.TickCount,两者之间没有根本区别。启动计时器时将值存储在变量中,在事件或回调中减去它以了解实际耗时。它在很长一段时间内都非常精确,精度受时钟更新速率的限制。默认情况下是每秒 64 次(15.6 毫秒),它可以通过调用 timeBeginPeriod() 来提升。 .保守地这样做,这对功耗不利。
秒表具有相反的特性,它高度准确但不精确。它未校准为像 DateTime.UtcNow 和 Environment.TickCount 这样的原子钟。并且对于生成事件完全没有用,在循环中燃烧核心以检测耗时将使您的线程在它的量子燃烧后被放入狗屋一段时间。仅将其用于临时分析。
关于c# - 结合 Stopwatch 和 Timer 以通过平滑强制获得长期准确的事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33000567/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!