- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我仍在努力让我的 FxCop 规则发挥作用。
作为其中的一部分,我需要弄清楚一个方法调用了哪些方法。以前我使用的是 CallGraph.CallersFor()
(反向操作,无论如何这是我的最终目标),但它似乎与我在下面描述的问题相同。
作为使用 CallGraph
类的替代方案,我尝试访问所有方法调用以基于此代码构建字典:
public override void VisitMethodCall(MethodCall call)
{
Method CalledMethod = (call.Callee as MemberBinding).BoundMember as Method;
// ....
}
但是,事实证明,如果被调用的方法是在重写基类方法的派生类上,则 BoundMember
是基类方法,而不是子类方法 (这是实际将被调用的那个)。
问题:如何获取在 FxCop 中的 callvirt IL 指令情况下将调用的方法?
最佳答案
事实证明,在我的情况下,我实际上并不完全需要这个(这很好,因为在这种情况下我没有确切的答案)。
因为 FxCop 是一个静态检查器,它永远无法知道变量指向的对象实例的类型,它可以声明为基类型。因此我相信我要求的是不可能的。
我的解决方案是,在构建调用树时,我为“调用”任何派生类的基类添加额外的引用。这样,如果我在基类上调用一个方法,并且可以调用派生类的方法,我可以按照调用树的方式进行调用。
请参阅下面的 FxCop 规则类使用的类:
public class CallGraphBuilder : BinaryReadOnlyVisitor
{
public Dictionary<TypeNode, List<TypeNode>> ChildTypes;
public Dictionary<Method, List<Method>> CallersOfMethod;
private Method _CurrentMethod;
public CallGraphBuilder()
: base()
{
CallersOfMethod = new Dictionary<Method, List<Method>>();
ChildTypes = new Dictionary<TypeNode, List<TypeNode>>();
}
public override void VisitMethod(Method method)
{
_CurrentMethod = method;
base.VisitMethod(method);
}
public void CreateTypesTree(AssemblyNode Assy)
{
foreach (var Type in Assy.Types)
{
if (Type.FullName != "System.Object")
{
TypeNode BaseType = Type.BaseType;
if (BaseType != null && BaseType.FullName != "System.Object")
{
if (!ChildTypes.ContainsKey(BaseType))
ChildTypes.Add(BaseType, new List<TypeNode>());
if (!ChildTypes[BaseType].Contains(Type))
ChildTypes[BaseType].Add(Type);
}
}
}
}
public override void VisitMethodCall(MethodCall call)
{
Method CalledMethod = (call.Callee as MemberBinding).BoundMember as Method;
AddCallerOfMethod(CalledMethod, _CurrentMethod);
Queue<Method> MethodsToCheck = new Queue<Method>();
MethodsToCheck.Enqueue(CalledMethod);
while (MethodsToCheck.Count != 0)
{
Method CurrentMethod = MethodsToCheck.Dequeue();
if (ChildTypes.ContainsKey(CurrentMethod.DeclaringType))
{
foreach (var DerivedType in ChildTypes[CurrentMethod.DeclaringType])
{
var DerivedCalledMethod = DerivedType.Members.OfType<Method>().Where(M => MethodHidesMethod(M, CurrentMethod)).SingleOrDefault();
if (DerivedCalledMethod != null)
{
AddCallerOfMethod(DerivedCalledMethod, CurrentMethod);
MethodsToCheck.Enqueue(DerivedCalledMethod);
}
}
}
}
base.VisitMethodCall(call);
}
private void AddCallerOfMethod(Method CalledMethod, Method CallingMethod)
{
if (!CallersOfMethod.ContainsKey(CalledMethod))
CallersOfMethod.Add(CalledMethod, new List<Method>());
if (!CallersOfMethod[CalledMethod].Contains(CallingMethod))
CallersOfMethod[CalledMethod].Add(CallingMethod);
}
private bool MethodHidesMethod(Method ChildMethod, Method BaseMethod)
{
while (ChildMethod != null)
{
if (ChildMethod == BaseMethod)
return true;
ChildMethod = ChildMethod.OverriddenMethod ?? ChildMethod.HiddenMethod;
}
return false;
}
}
关于c# - 如何获取FxCop中callvirt IL指令实际调用的方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6454181/
我试图了解 CLR 如何实现引用类型和多态性。我提到了 Don Box 的 Essential .Net Vol 1,它对校准大部分内容有很大帮助。但是当我尝试使用一些 IL 代码以更好地理解时,我被
CIL 指令“Call”和“Callvirt”有什么区别? 最佳答案 当运行时执行call指令时,它会调用一段确切的代码(方法)。毫无疑问它存在于何处。 IL 经过 JIT 处理后,调用站点生成的机器
是 il.EmitCall(OpCodes.Callvirt, getter, null); 等同于: 到 il.Emit(OpCodes.Callvirt,getter); 我不确定如何检查动态生成
如何确定一个方法是否需要使用“Call”或“Callvirt”来调用? 最佳答案 您可以一一遵循这些简单的规则来确定您应该使用哪一个: 方法是静态的吗?然后使用调用。 您正在调用的类型是值类型吗?然后
这个问题链接到How CLR calls methods correctly hidden by derived class when reference is stored in a base cl
在这种情况下,IL 并不总是对 virtual 方法使用 callvirt 指令: class MakeMeASandwich{ public override string ToString()
向某人解释虚拟调度很容易:每个对象都有一个指向表的指针作为其数据的一部分。类上有N个虚方法。每次调用特定方法时,我都会在对象到达时对其进行索引,并调用表中的第 i 个方法。每个实现方法 X() 的类都
在 IL 的一些实验中,我试图将程序集中的 callvirt 调用更改为 call 方法。基本上发生的事情是我有一个继承链,其中包含我正在调用的成员函数。 基本上调用与此类似: ((MyDerived
给定两种方法: static void M1(Person p) { if (p != null) { var p1 = p.N
对于下面的代码片段: struct Test { public override string ToString() { return ""; } } publ
我在其基类中标记为抽象的库类上调用属性集访问器。现在在运行时我 force应用程序针对另一个版本的库运行,其中类仅实现基类的底层接口(interface),但不是从它派生的。 有趣的是,.NET 将运
为什么编译器生成一个 callvirt 指令用于调用显式实现的接口(interface)方法和一个 call 用于调用 implicilty 在以下代码中实现了接口(interface)方法? 编译器
我有几个月前编写的这个框架,它生成一个类来调用此性能服务。框架的消费者使用方法创建一个接口(interface),使用属性进行注释,并调用一个工厂方法来创建他们可以用来调用此性能服务的接口(inter
我很想知道为什么会这样。请阅读下面的代码示例以及在每个部分下面的注释中发出的相应 IL: using System; class Program { static void Main()
我有一个类: public class SomeClass { public int I; public SomeClass(int input) { I = inpu
IL 提供了两种调用函数的语句,即 call 和 callvirt。 Call 用于调用非虚拟或静态函数或编译器不想对引用进行空检查的任何函数。 callvirt 用于调用虚函数,非虚函数也被调用,因
我有一个递归函数 emit : Map -> exp -> unit哪里il : ILGenerator对函数来说是全局的,并且 exp是一个判别联合,表示带有大小写 InstanceCall of
我是一名优秀的程序员,十分优秀!