如何确定方法中调用哪些方法?

时间:2010-11-01 23:04:33

标签: c# .net mono.cecil

我想列出从特定方法调用的所有方法。例如。如果我有以下代码:

public void test1() {

   test2();


   test3();
}

该列表应包含test2()和test3()。如果可以列出相同类的方法以及另一个类的方法,那将是很好的。

另外我想找到一种方法来检测方法使用哪些字段:

public class A {

   private String test1 = "";
   private String test2 = "";

   public void test() {
      Console.WriteLine(test1);
   }

}

因此应该列出test1。

我尝试使用Mono.Cecil,但遗憾的是我找不到很多关于该项目的文档。那么有人知道怎么做吗?

编辑:我想用Mono.Cecil来做,因为通过它的API,我可以直接在我的应用程序中使用结果。如果我在Visual Studio或类似工具中使用内置工具,则很难处理结果。

3 个答案:

答案 0 :(得分:7)

我没有真正使用过Cecil,但是HowTo页面显示了如何枚举类型,你的问题似乎只需要循环你的后续指令:Call和Load Field。这个示例代码似乎处理了您提到的案例,但可能还有更多,您应该检查其他调用说明。如果你使它递归,请确保你跟踪你已经检查过的方法。

static void Main(string[] args)
{
    var module = ModuleDefinition.ReadModule("CecilTest.exe");

    var type = module.Types.First(x => x.Name == "A");
    var method = type.Methods.First(x => x.Name == "test");

    PrintMethods(method);
    PrintFields(method);

    Console.ReadLine();
}

public static void PrintMethods(MethodDefinition method)
{
    Console.WriteLine(method.Name);
    foreach (var instruction in method.Body.Instructions)
    {
        if (instruction.OpCode == OpCodes.Call)
        {
            MethodReference methodCall = instruction.Operand as MethodReference;
            if(methodCall != null)
                Console.WriteLine("\t" + methodCall.Name);
        }
    }
}


public static void PrintFields(MethodDefinition method)
{
    Console.WriteLine(method.Name);
    foreach (var instruction in method.Body.Instructions)
    {
        if (instruction.OpCode == OpCodes.Ldfld)
        {
            FieldReference field = instruction.Operand as FieldReference;
            if (field != null)
                Console.WriteLine("\t" + field.Name);
        }
    }
}

答案 1 :(得分:1)

仅使用C#中的反射API无法做到这一点。真的,您需要解析原始源代码,这可能不是您正在寻找的解决方案。但是,例如,这就是Visual Studio如何通过这种信息进行重构。

你可能会在某处分析IL - 就像Reflector所做的那样,但我认为这将是一项巨大的工作。

答案 2 :(得分:1)

如果您想付费,可以使用.NET Reflector tool。你也可以看一下这个.NET Method Dependencies它会变得棘手,因为你将要进入IL。第三种可能是在VS中使用宏引擎,它确实有分析代码的工具,CodeElement,我不确定它是否可以做依赖。