如何跟踪调用的每个方法

本文关键字:方法 调用 何跟踪 跟踪 | 更新日期: 2023-09-27 17:59:26

我有一个现有的项目,我想找出正在进行的所有调用,并可能转储到日志文件中。

我看了一下这根线,但帮不了什么忙。我尝试了PostSharp,这个例子展示了如何实现它。但我需要为每个该死的方法添加一个属性。作为一个现有的项目,在许多方法中,这是不可行的选择。

有没有其他方法可以让我快速追踪所有拨打的电话?

如何跟踪调用的每个方法

您可以使用Unity Interception 进行此操作

请参阅本文以获取示例。本文使用了属性,但我下面的代码示例使用依赖注入系统(对接口进行编码)来设置拦截。

如果你想记录MyClass,它是这样的:

  1. 制作一个包含MyClass=>IMyClass中所有方法的接口
  2. 您可以设置InterfaceInterception(就像我在下面所做的那样),或者还有其他一些方法可以设置它。请参阅此处了解所有选项
  3. 您将设置一个策略来拦截所有与IMatchingRule匹配的方法
  4. 所有调用现在都将被ICallHandler实现截获

代码:

//You  will use the code like this:
MyContainer container = new MyContainer();
//setup interception for this type..
container.SetupForInteception(typeof(IMyClass));
 //what happens here is you get a proxy class 
 //that intercepts every method call.
IMyClass cls = container.Resolve<IMyClass>();
 //You need the following for it to work:   
public class MyContainer: UnityContainer
{
    public MyContainer()
    {
        this.AddNewExtension<Interception>();
        this.RegisterType(typeof(ICallHandler), 
                    typeof(LogCallHandler), "MyCallHandler");
        this.RegisterType(typeof(IMatchingRule), 
                       typeof(AnyMatchingRule), "AnyMatchingRule");
        this.RegisterType<IMyClass, MyClass>();
    }
    //apparently there is a new way to do this part
    // http://msdn.microsoft.com/en-us/library/ff660911%28PandP.20%29.aspx
    public void SetupForInteception(Type t)
    {
        this.Configure<Interception>()
        .SetInterceptorFor(t, new InterfaceInterceptor())
        .AddPolicy("LoggingPolicy")
        .AddMatchingRule("AnyMatchingRule")
        .AddCallHandler("MyCallHandler");
    }
}
//THIS will match which methods to log.
public class AnyMatchingRule : IMatchingRule
{
    public bool Matches(MethodBase member)
    {
        return true;//this ends up loggin ALL methods.
    }
}
public class LogCallHandler : ICallHandler
{
    public IMethodReturn 
             Invoke(IMethodInvocation input, GetNextHandlerDelegate getNext)
    {
      //All method calls will result in a call here FIRST.
      //IMethodInvocation has an exception property which will let you know
      //if an exception occurred during the method call.
    }
 }

在跟踪模式下使用档案器。然后你会看到每件事都是如何相互称呼的,以及时间花在了哪里。除了商业评测器,还有免费的。对于托管代码,有一个NPProfiler,它非常好。

如果你想更深入,你可以使用Windows性能工具包,它可以为你提供所有线程的完整信息,以及如果你想知道的话,它们是如何相互交互的。唯一的区别是,您得到的堆栈范围从内核到托管帧。

如果这还不够,您可以使用跟踪库(使用PostSharp自动…)或手动或使用每个源文件的宏来插入代码。我制作了一个小的跟踪库,它非常快速且高度可配置。请参见此处。作为唯一的功能,它可以自动跟踪任何抛出的异常。

private void SomeOtherMethod()
{
  using (Tracer t = new Tracer(myType, "SomeOtherMethod"))
  {
      FaultyMethod();
  }
}
private void FaultyMethod()
{
   throw new NotImplementedException("Hi this a fault");
}

输出如下:

    18:57:46.665  03064/05180 <{{         > ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeMethod  
    18:57:46.668  03064/05180 <{{         > ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeOtherMethod  
    18:57:46.670  03064/05180 <         }}< ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeOtherMethod Exception thrown: System.NotImplementedException: Hi this a fault    
at ApiChange.IntegrationTests.Diagnostics.TracingTests.FaultyMethod()  
at ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeOtherMethod()  
at ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeMethod()    
at ApiChange.IntegrationTests.Diagnostics.TracingTests.Demo_Show_Leaving_Trace_With_Exception() 
18:57:46.670  03064/05180 <         }}< ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeOtherMethod Duration 2ms 18:57:46.689  03064/05180 <         }}< ApiChange.IntegrationTests.Diagnostics.TracingTests.SomeMethod Duration 24ms
PostSharp当然提供了一种将方面应用于多个目标的方法,而无需显式地用属性装饰它们。请参见多播属性。

当开发(多播)方面时,您必须指定其用法:

[MulticastAttributeUsage(MulticastTargets.Method, TargetMemberAttributes = MulticastAttributes.Instance)]
[AttributeUsage(AttributeTargets.Assembly|AttributeTargets.Class|AttributeTargets.Method, AllowMultiple = true)]
[Serializable]
public class TraceAttribute : MethodInterceptionAspect
{
// Details skipped.
}

然后以覆盖用例的方式应用方面(例如AdventureWorks.BusinessLayer命名空间中的所有公共成员):

[assembly: Trace( AttributeTargetTypes="AdventureWorks.BusinessLayer.*", AttributeTargetMemberAttributes = MulticastAttributes.Public )]