从方法中检索调用方法名称

Sco*_*ski 57 c# reflection methods calling-convention

可能重复:
如何找到调用当前方法的方法?

我在对象中有一个方法,从对象中的许多地方调用.是否有一种快速简便的方法来获取调用此流行方法的方法的名称.

伪代码示例:

public Main()
{
     PopularMethod();
}

public ButtonClick(object sender, EventArgs e)
{
     PopularMethod();
}

public Button2Click(object sender, EventArgs e)
{
     PopularMethod();
}

public void PopularMethod()
{
     //Get calling method name
}
Run Code Online (Sandbox Code Playgroud)

在内部,PopularMethod()我希望看到Main它是否被调用的价值Main......我想看看" ButtonClick"是否PopularMethod()被调用ButtonClick

我在看,System.Reflection.MethodBase.GetCurrentMethod()但这不会让我得到调用方法.我看过这个StackTrace类,但是每次调用该方法时我都不喜欢运行整个堆栈跟踪.

Mar*_*ell 156

在.NET 4.5/C#5中,这很简单:

public void PopularMethod([CallerMemberName] string caller = null)
{
     // look at caller
}
Run Code Online (Sandbox Code Playgroud)

编译器自动将来电者的姓名; 所以:

void Foo() {
    PopularMethod();
}
Run Code Online (Sandbox Code Playgroud)

会传入"Foo".

  • @Feckmore无论如何:编译器在编译期间将其添加为文字.它会明显快于看StackTrace等等.基本上它被编译为PopularMethod("CurrentMethodName") (5认同)
  • 该解决方案的性能后果是什么? (3认同)

jas*_*son 72

我不认为没有跟踪堆栈就可以完成.但是,这样做很简单:

StackTrace stackTrace = new StackTrace();
MethodBase methodBase = stackTrace.GetFrame(1).GetMethod();
Console.WriteLine(methodBase.Name); // e.g.
Run Code Online (Sandbox Code Playgroud)

但是,我认为你真的必须停下来问自己是否有必要.

  • System.Diagnostics是StackTrace的命名空间,System.Reflection是MethodBase的命名空间. (3认同)
  • @Scott Vercuski - 这是一个非常昂贵的操作,如果有任何方法可以避免这种情况,你真的应该这样做.它比抛出异常更糟糕,这很糟糕.它可以做到很好,但它不应该被滥用. (2认同)

Joh*_*ren 17

这其实很简单.

public void PopularMethod()
{
    var currentMethod = System.Reflection.MethodInfo
        .GetCurrentMethod(); // as MethodBase
}
Run Code Online (Sandbox Code Playgroud)

但要小心,如果内联方法有任何影响,我有点怀疑.您可以这样做以确保JIT编译器不会妨碍您.

[System.Runtime.CompilerServices.MethodImpl(
 System.Runtime.CompilerServices.MethodImplOptions.NoInlining)]
public void PopularMethod()
{
    var currentMethod = System.Reflection.MethodInfo
        .GetCurrentMethod();
}
Run Code Online (Sandbox Code Playgroud)

要获得调用方法:

[System.Runtime.CompilerServices.MethodImpl(
 System.Runtime.CompilerServices.MethodImplOptions.NoInlining)]
public void PopularMethod()
{
    // 1 == skip frames, false = no file info
    var callingMethod = new System.Diagnostics.StackTrace(1, false)
         .GetFrame(0).GetMethod();
}
Run Code Online (Sandbox Code Playgroud)

  • John Leidgren - 这将命名为"PopularMethod",但我想要名为PopularMethod的方法的名称. (3认同)

Sru*_*uly 5

只需传入参数即可

public void PopularMethod(object sender)
{

}
Run Code Online (Sandbox Code Playgroud)

国际海事组织:如果它对事件来说足够好,它应该足够好了.

  • 我为许多系统编写了代码.开发人员通常希望跟踪编写日志条目的方法.如果日志记录代码需要手动传递方法名称,则这要求开发人员在每次调用logging方法时都将方法名称作为文本输入.现在在程序的许多不同部分使用相同的日志记录方法,并查看在几年内与许多开发人员一起添加,删除和重命名方法时会发生什么.自动传递调用者的方法名称将极大地简化开发和调试. (2认同)

小智 5

我经常发现自己想要这样做,但最终总是重构我的系统设计,这样我就不会出现这种“尾巴摇狗”的反模式。结果始终是一个更健壮的架构。