微信公众号“dotNET跨平台”的文章《一个监控C#方法运行耗时开源库》介绍了支持测量方法耗时的包MethodTimer.Fody,使用方便,还可以自定义输出信息格式。本文学习并测试MethodTimer.Fody包的使用方式。
新建控制台程序,通过Nuget包管理器添加MethodTimer.Fody和Fody(参考文献1和3中都提到需要安装这两个包,但是测试时仅安装MethodTimer.Fody也可以正常运行,暂不清楚怎么回事)。
MethodTimer.Fody库支持两种耗时时间形式:1)时段,TimeSpan类型;2)总耗毫秒数,long类型。
采用MethodTimer.Fody库计算方法耗时时,在需计算耗时的方法上赋予Time属性,同时定义拦截器(intercepter)获取方法耗时信息,既可以将信息输出到控制台,也可以按需输出到文件或其它地方。
以开源博客项目Blog中的雪花生成算法类为例,计算SnowflakeId.NextStringId()函数的耗时,其设置形式为:
[Time("生成雪花ID测试")]
public static string NextStringId()
同时定义拦截器截获并输出耗时信息,可以输出时段,也可按总毫秒数输出:
public static class MethodTimeLogger
{//按时段输出public static void Log(MethodBase methodBase, TimeSpan elapsed, string message){Console.WriteLine($"方法:{methodBase.Name} 耗时(时段):{elapsed}, 信息:{message}");}//按总毫秒数输出public static void Log(MethodBase methodBase, long milliseconds, string message){Console.WriteLine($"方法:{methodBase.Name} 耗时(总毫秒数):{milliseconds}, 信息:{message}");}
}
从上面的耗时来看,Blog项目中的雪花算法运行速度很快,但距离其极限速度还差距较大(项目的注释中介绍SnowFlake单机每秒都能够产生出极限4,096,000个ID)。Time特性中添加的文本会一并输出到控制台中
根据MethodTimer.Fody库的GitHub主页说明,给方法添加Time特性,编译时会将函数主体置于System.Diagnostics.Stopwatch的StartNew和Stop函数之间,以计算函数耗时,如果没有定义拦截器,模式将耗时信息调用Trace.WriteLine输出,定义拦截器后,则调用拦截器的函数输出耗时信息(测试时没有定义拦截器的话,程序编译会报错)。
Time特性除了用于函数,也可以用于构造函数、类、模块或程序集,将Time定义在SnowflakeId类上,则调用NextStringId函数时会输出所有调用过的函数的耗时。
[AttributeUsage(AttributeTargets.Assembly | AttributeTargets.Module | AttributeTargets.Class | AttributeTargets.Constructor | AttributeTargets.Method)]public class TimeAttribute : Attribute
参考文献:
[1]https://blog.csdn.net/sD7O95O/article/details/134622718
[2]https://github.com/Fody/Fody
[3]https://github.com/Fody/MethodTimer