使用StackTrace

时间:2018-06-24 15:00:35

标签: c# reflection system.reflection

我有两种情况想知道哪个是我正在执行的某些方法的调用者方法。

这些是方案:

1)

public static void ExecuteMethod(object obj)
{
    var mth = new StackTrace().GetFrame(1).GetMethod();
    string methodName = mth.Name;
}

我这样称呼:

public class Process
{
    public int Handle { get; set; }
    public string Name { get; set; }
    public int ProcessID { get; set; }

    public dynamic GetOwner()
    {
        return WMIMethod.ExecuteMethod(this);
    }
}

执行此操作时,methodName的结果是我期望的结果:GetOwner

第二个有问题的情况是这种情况:

2)

public static dynamic ExecuteMethod(object obj, dynamic parameters)
{
    var mth = new StackTrace().GetFrame(1).GetMethod();
    string methodName = mth.Name;
}

我这样称呼:

public class Process
{
    public int Handle { get; set; }
    public string Name { get; set; }
    public int ProcessID { get; set; }

    public dynamic GetOwner(dynamic inParams)
    {
        return WMIMethod.ExecuteMethod(this, inParams);
    }
}

在这种情况下,当我打电话给new Process().GetOwner(new { MyParam = "Something" } )时,methodName的结果不再是我期望的(GetOwner)了,取而代之的是methodName CallSite.Target,而mth的结果是

{Void CallSite.Target(System.Runtime.CompilerServices.Closure, System.Runtime.CompilerServices.CallSite, System.Type, ORMi.Sample.Models.Printer, System.Object)}

任何人都知道第二种情况为何与第一种情况不同?这怎么解决??

谢谢!。

1 个答案:

答案 0 :(得分:1)

根据我的发现,当使用dynamic对象时,C#添加了一个额外的方法调用System.Dynamic.UpdateDelegates.UpdateAndExecute3()

在您的情况下,我将第二种方法重写为

public static dynamic ExecuteMethod(object obj, dynamic p)
{
    var frame =
        new StackTrace().GetFrames()
                        .Skip(1) // Skip the 'ExecuteMethod'
                        .First(x => x.GetMethod().DeclaringType.Namespace != "System.Dynamic");

    return frame.GetMethod().Name;
}

不幸的是,我不知道为什么C#会插入该调用,因此,如果有人可以解释该内部工作,我将不胜感激。