.net平台下获取高精度时间类

时间:2023-03-08 22:58:11
.net平台下获取高精度时间类

原文:http://www.it118.org/specials/321869dd-98cb-431b-b6d2-82d973cd739d/d940a15c-0d36-4369-8de0-b07cf3f3fd5f.htm

前言:.NET 2.0前运行库中不存在高精度的计时器,而您又需要它,解决的方法是通过调用QueryPerformanceFrequency 和 QueryPerformanceCounter这两个Win32 API来实现。在.NET 2.0时,stopwatch类也可实现高精度计时,请参考:《使用Stopwatch类实现高精度计时

.NET运行库具有一个属性—— System.Environment.Tickcount,您可以使用它计算时间。该属性返回计算机最近一次启动之后,已用时间滴答数(ms)。一开始,您可能觉得它是一个非常好的工具。但是,它有一个相当显著的缺陷,即该属性并不是每毫秒都更新。

该属性的更新频率通常依赖于计时器的分辨率。在该案例中,tick-count属性的分辨率是15 ms。如果您在一个循环中持续访问该属性,则在更新之前的每15ms它返回相同的值,然后在下一次更新之前的另一个15ms中,它返回一个新的值。现代的计算机每秒能够执行非常巨大的计算量,15ms的分辨率将使得您的计算看上去好像“停停动动的”。

警告:

如果希望在应用程序中使用TickCount属性,那么必须明确它返回的是一个无符号整数值。因为该属性是计算机启动之后已用时间的滴答数,所以如果计算机运行了非常长的时间 (例如超过25天),该属性将返回一个负数。如果您没有考虑这种情况,将使得正在使用的公式出错。在更长的一段时间之后,该值将重新归0。

这里您需要的是一个具有更高分辨率的计时器。一个分辨率为1ms的计时器是您非常好的选择。

.NET 2.0前运行库中不存在高精度的计时器,而您又需要它,解决的方法是通过调用QueryPerformanceFrequency 和 QueryPerformanceCounter这两个Win32 API来实现。

  1. using System;
  2. using System.Runtime.InteropServices;
  3. internal sealed class AccurateTimerHelper
  4. {
  5. public AccurateTimerHelper()
  6. {
  7. TicksPerSecond = this.GetTicksPerSecond();
  8. this.BaseTime = this.GetTime();
  9. }
  10. [DllImport("kernel32.dll", EntryPoint="QueryPerformanceCounter")]
  11. private static extern int QueryPerformanceCounter(ref long lpPerformanceCount);
  12. [DllImport("kernel32.dll", EntryPoint="QueryPerformanceFrequency")]
  13. private static extern int QueryPerformanceFrequency(ref long A_0);
  14. public long GetElapsedTime()
  15. {
  16. return (long)((((double)(this.GetTime() - this.BaseTime)) / ((double)TicksPerSecond)) * 10000);
  17. }
  18. private long GetTime()
  19. {
  20. long time = 0;
  21. if (AccurateTimerHelper.QueryPerformanceCounter(ref time) == 0)
  22. {
  23. throw new NotSupportedException("Error while querying the high-resolution performance counter.");
  24. }
  25. return time;
  26. }
  27. private long GetTicksPerSecond()
  28. {
  29. long ticksPerSecond = 0;
  30. if (AccurateTimerHelper.QueryPerformanceFrequency(ref ticksPerSecond) == 0)
  31. {
  32. throw new NotSupportedException("Error while querying the performance counter frequency.");
  33. }
  34. return ticksPerSecond;
  35. }
  36. private long BaseTime
  37. {
  38. get
  39. {
  40. return baseTime;
  41. }
  42. set
  43. {
  44. this.baseTime = value;
  45. }
  46. }
  47. private long TicksPerSecond
  48. {
  49. get
  50. {
  51. return this.ticksPerSecond;
  52. }
  53. set
  54. {
  55. ticksPerSecond = value;
  56. }
  57. }
  58. private long baseTime;
  59. private long ticksPerSecond;
  60. }

调用很简单:

    1. AccurateTimerHelper atHelper = new AccurateTimerHelper();
    2. /*
    3. //进行很多费时操作
    4. */
    5. atHelper.GetElapsedTime()