• (转)C#精确时间计时器


    原文地址:http://blog.sina.com.cn/s/blog_699d3f1b01012vgb.html

    1 调用WIN API中的GetTickCount

    [DllImport("kernel32")]
    static extern uint GetTickCount();

    从操作系统启动到现在所经过的毫秒数,精度为1毫秒,经简单测试发现其实误差在大约在15ms左右

    缺点:返回值是uint,最大值是2的32次方,因此如果服务器连续开机大约49天以后,该方法取得的返回值会归零

    用法:

    1
    2
    3
    uint s1 = GetTickCount();
    Thread.Sleep(2719);
    Console.WriteLine(GetTickCount() - s1);  //单位毫秒

    2 调用WIN API中的timeGetTime  推荐

    [DllImport("winmm")]
    static extern uint timeGetTime();

    常用于多媒体定时器中,与GetTickCount类似,也是返回操作系统启动到现在所经过的毫秒数,精度为1毫秒。

    一般默认的精度不止1毫秒(不同操作系统有所不同),需要调用timeBeginPeriod与timeEndPeriod来设置精度

    [DllImport("winmm")]
    static extern void timeBeginPeriod(int t);
    [DllImport("winmm")]
    static extern void timeEndPeriod(int t);

     缺点:与GetTickCount一样,受返回值的最大位数限制。

    用法:

    1
    2
    3
    4
    5
    timeBeginPeriod(1);
    uint start = timeGetTime();
    Thread.Sleep(2719);
    Console.WriteLine(timeGetTime() - start);  //单位毫秒
    timeEndPeriod(1);

     3 调用.net自带的方法System.Environment.TickCount

    获取系统启动后经过的毫秒数。经反编译猜测它可能也是调用的GetTickCount,但是它的返回值是int,而GetTickCount与 timeGetTime方法的原型中返回值是DWORD,对应C#中的uint,难道.NET对 System.Environment.TickCount另外还做了什么处理么?
    缺点:与GetTickCount一样,受返回值的最大位数限制。

    用法:

    1
    2
    3
    int aa = System.Environment.TickCount;
    Thread.Sleep(2719);
    Console.WriteLine(System.Environment.TickCount - aa); //单位毫秒

     

     :经过测试,发现GetTickCount、System.Environment.TickCount也可以用timeBeginPeriod与timeEndPeriod来设置精度,最高可将精度提高到1毫秒。不知是什么原因?

     4 调用WIN API中的QueryPerformanceCounter

    [DllImport("kernel32.dll ")]
    static extern bool QueryPerformanceCounter(ref long lpPerformanceCount);

    用于得到高精度计时器(如果存在这样的计时器)的值。微软对这个API解释就是每秒钟某个计数器增长的数值。
    如果安装的硬件不支持高精度计时器,函数将返回false需要配合另一个API函数QueryPerformanceFrequency。

    [DllImport("kernel32")]
    static extern bool QueryPerformanceFrequency(ref long PerformanceFrequency);

     QueryPerformanceFrequency返回硬件支持的高精度计数器的频率,如果安装的硬件不支持高精度计时器,函数将返回false。

    用法:

    1
    2
    3
    4
    5
    6
    7
    long a = 0;
    QueryPerformanceFrequency(ref a);
    long b = 0, c = 0;
    QueryPerformanceCounter(ref b);
    Thread.Sleep(2719);
    QueryPerformanceCounter(ref c);
    Console.WriteLine((c - b) / (decimal)a);  //单位秒

    精度为百万分之一秒。而且由于是long型,所以不存在上面几个API位数不够的问题。

    缺点:在一篇文章看到,该API在节能模式的时候结果偏慢,超频模式的时候又偏快,而且用电池和接电源的时候效果还不一样(笔记本)
    原文地址:http://delphi.xcjc.net/viewthread.php?tid=1570
    未经过超频等测试,如果是真的,那该API出来的结果就可能不准。

     5 使用.net的System.Diagnostics.Stopwatch类    推荐

    Stopwatch 在基础计时器机制中对计时器的刻度进行计数,从而测量运行时间。如果安装的硬件和操作系统支持高分辨率性能的计数器,则 Stopwatch 类将使用该计数器来测量运行时间;否则,Stopwatch 类将使用系统计数器来测量运行时间。使用 Frequency 和 IsHighResolution 两个静态字段可以确定实现 Stopwatch 计时的精度和分辨率。

    实际上它里面就是将QueryPerformanceCounter、QueryPerformanceFrequency两个WIN API封装了一下,如果硬件支持高精度,就调用QueryPerformanceCounter,如果不支持就用DateTime.Ticks来计算。

    用法:

    Stopwatch sw = new Stopwatch();
    sw.Start();
    Thread.Sleep(2719);
    sw.Stop();
    Console.WriteLine(sw.ElapsedTicks / (decimal)Stopwatch.Frequency);
     

    6 用 DateTime.Now.Ticks

         http://www.cnblogs.com/csharp4/archive/2010/07/24/1784094.html

           1个 tick 代表100纳秒,1毫秒等于10000个ticks,也就是说,1秒等于 1E7 个ticks。

    任何一个 DateTime 型变量中都有一个 ticks 属性,它代表从 12:00:00 midnight, January 1, 0001 到当前时间值的间隔的 ticks 值。

    用上面的方法,最精确,应该也就是精确到1tick,即100纳秒。


  • 相关阅读:
    3里氏代换原则LSP
    2单一职责原则SRP
    1开放封闭原则OCP
    24访问者模式Visitor
    python json模块,处理json文件的读写
    python zip 绑定多个list
    python 字符串重复多次的技巧 *操作符
    python 刷新缓冲区,实时监测
    python os.getcwd 获取工作目录
    python datetime 获取时间
  • 原文地址:https://www.cnblogs.com/hhhh2010/p/4168278.html
Copyright © 2020-2023  润新知