如何使用C# Stopwatch 测量微秒级精确度

 更新时间:2021年03月04日 09:44:44   作者:黑暗执行绪  
这篇文章主要介绍了如何使用C# Stopwatch 测量微秒精确度,帮助大家更好的理解和学习使用c#,感兴趣的朋友可以了解下

跟同事讨论到- 用C# Stopwatch 取得效能数值,Stopwatch.ElapsedMilliseconds 只到毫秒(ms),如果需要更高的时间精确度(微秒μs,甚至奈秒ns),该怎么做?

原以为要费番功夫,在Stackoverlow查到讨论,答案意外地简单。

准备测试程式如下,比较MD5 及SHA1 计算1MB byte[] 杂凑值所秏费时间:

static byte[] data = new byte[1024 * 1024];
static void Main(string[] args)
{
  Test1();
  Console.ReadLine();
}

private static void Test1()
{
  Console.WriteLine("Test 1");
  for (var i = 0; i < 5; i++)
  {
    Stopwatch sw = new Stopwatch();
    sw.Start();
    var md5 = MD5.Create().ComputeHash(data);
    sw.Stop();
    Console.WriteLine($"MD5 {sw.ElapsedMilliseconds}ms");
    sw.Restart();
    var sha1 = SHA1.Create().ComputeHash(data);
    sw.Stop();
    Console.WriteLine($"SHA1 {sw.ElapsedMilliseconds}ms");
  }
}

执行结果如下:

Test 1
MD5 10ms
SHA1 2ms
MD5 2ms
SHA1 2ms
MD5 2ms
SHA1 2ms
MD5 2ms
SHA1 2ms
MD5 2ms
SHA1 2ms

有两个问题,第一是回圈的第一次执行因涉及.NET 初始化,耗时会异常偏高(先做SHA1 再做MD5,就变成第一笔SHA1 超过10ms),第二是MD5 与SHA1 执行时间相近,都是2ms 多,用ElapsedMilliseconds 看不出差异。

针对首次数值耗时偏差问题,除了略过第一次数据不计,我想到的另一个解法是在Test1()前先跑一次MD5.Create()完成相关初始化。至于ElapsedMilliseconds看不出差异问题,改用ElapsedTicks是种解法,但要注意,ElaspedTicks换算成时间单位时,不是除以TimeSpan.TicksPerMillisecond而是依CPU频率而定,需使用Stopwatch.Frequency (每秒Tick数)。

第二版改用ElapsedTicks * 1000000F / Stopwatch.Frequency 计算微秒(Microsecond, μs),执行前先MD5.Create() 暖机。

static byte[] data = new byte[1024 * 1024];
static void Main(string[] args)
{
  MD5.Create();
  Test2();
  Console.ReadLine();
}

private static void Test2()
{
  Console.WriteLine("Test 2");
  for (var i = 0; i < 5; i++)
  {
    Stopwatch sw = new Stopwatch();
    sw.Start();
    var md5 = MD5.Create().ComputeHash(data);
    sw.Stop();
    //
    Console.WriteLine($"MD5 {sw.ElapsedTicks * 1000000F / Stopwatch.Frequency:n3}μs");
    sw.Restart();
    var sha1 = SHA1.Create().ComputeHash(data);
    sw.Stop();
    Console.WriteLine($"SHA1 {sw.ElapsedTicks * 1000000F / Stopwatch.Frequency:n3}μs");
  }
}

执行结果的第一次时间偏长问题消失,而也呈现出SHA1 比MD5 计算耗时的证据。而由数值来看,精确度可到0.1μs = 100ns。

Test 2
MD5 2,402.200μs
SHA1 2,724.000μs
MD5 2,017.300μs
SHA1 2,576.900μs
MD5 2,102.100μs
SHA1 2,578.700μs
MD5 2,024.100μs
SHA1 2,600.300μs
MD5 2,008.300μs
SHA1 2,624.300μs

自己计算麻烦了点,Stopwatch 有个Elapsed 属性,型别为TimeSpan,其中TotalMilliseconds 属性精确度即可达到μs 及100ns。请看第三版:

    static byte[] data = new byte[1024 * 1024];
    static void Main(string[] args)
    {
      MD5.Create();
      Test3();
      Console.ReadLine();
    }
    private static void Test3()
    {
      Console.WriteLine("Test 3");
      for (var i = 0; i < 5; i++)
      {
        Stopwatch sw = new Stopwatch();
        sw.Start();
        var md5 = MD5.Create().ComputeHash(data);
        sw.Stop();
        Console.WriteLine($"MD5 {sw.Elapsed.TotalMilliseconds * 1000:n3}μs");
        sw.Restart();
        var sha1 = SHA1.Create().ComputeHash(data);
        sw.Stop();
        Console.WriteLine($"SHA1 {sw.Elapsed.TotalMilliseconds * 1000:n3}μs");
      }
    }

执行结果与第二版相同,但程式更简单一些。

Test 3
MD5 2,423.400μs
SHA1 2,692.400μs
MD5 2,204.000μs
SHA1 2,976.800μs
MD5 2,094.500μs
SHA1 2,588.600μs
MD5 2,034.600μs
SHA1 2,598.900μs
MD5 2,029.900μs
SHA1 2,887.000μs

以上就是如何使用C# Stopwatch 测量微秒精确度的详细内容,更多关于C# Stopwatch 测量微秒精确度的资料请关注脚本之家其它相关文章!

相关文章

  • C#中的timer与线程使用

    C#中的timer与线程使用

    这篇文章主要介绍了C#中的timer与线程使用方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • C# task应用实例详解

    C# task应用实例详解

    这篇文章主要介绍了如何在C#中一些应用task的实例,简单易懂的代码能更好的帮你学习,有兴趣的朋友可以了解下
    2020-05-05
  • C#定义简单的反射工厂实例分析

    C#定义简单的反射工厂实例分析

    这篇文章主要介绍了C#定义简单的反射工厂的用法,实例分析了反射工厂的原理与使用技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-04-04
  • C#使用PPT组件的CreateVideo方法实现视频生成

    C#使用PPT组件的CreateVideo方法实现视频生成

    这篇文章主要为大家详细介绍了C#如何使用PPT组件的CreateVideo方法实现视频生成,文中的示例代码讲解详细,有需要的小伙伴可以跟随小编一起学习一下
    2023-10-10
  • C#实现合并多张图片为GIF动态图

    C#实现合并多张图片为GIF动态图

    这篇文章主要为大家详细介绍了C#如何将把一张又一张的图片去拼合成一张GIF动态图片,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2022-12-12
  • Unity通用泛型单例设计模式(普通型和继承自MonoBehaviour)

    Unity通用泛型单例设计模式(普通型和继承自MonoBehaviour)

    这篇文章主要介绍了Unity通用泛型单例设计模式,分为普通型和继承MonoBehaviour,帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-07-07
  • 浅谈C# 类的继承

    浅谈C# 类的继承

    本文主要介绍了C# 类的继承相关知识。具有很好的参考价值,下面跟着小编一起来看下吧
    2017-01-01
  • C# winform 模拟键盘输入自动接入访问网络的实例

    C# winform 模拟键盘输入自动接入访问网络的实例

    本篇文章主要介绍了C# winform 模拟键盘输入自动接入访问网络,有兴趣的可以了解一下。
    2016-11-11
  • c#冒泡排序算法示例

    c#冒泡排序算法示例

    这篇文章主要介绍了c#冒泡排序算法示例,需要的朋友可以参考下
    2014-04-04
  • C# 6.0 的知识梳理

    C# 6.0 的知识梳理

    目前最新的版本是C# 7.0,VS的最新版本为Visual Studio 2017 RC,两者都尚未进入正式阶段。C# 6.0虽说出了一段时间,但是似乎有许多人对这一块知识并不了解。本文将对C# 6.0的知识进行梳理,下面跟着小编一起来看下吧
    2016-12-12

最新评论