确定一个方法是不是调用另一个包含新语句的程序集中的方法,反之亦然

Posted

技术标签:

【中文标题】确定一个方法是不是调用另一个包含新语句的程序集中的方法,反之亦然【英文标题】:Determining if a method calls a method in another assembly containing a new statement and vice-versa确定一个方法是否调用另一个包含新语句的程序集中的方法,反之亦然 【发布时间】:2011-06-22 11:54:26 【问题描述】:

我想编写一个规则,如果在由标记有特定属性的方法调用的任何方法中进行对象分配,该规则将失败。

到目前为止,我已经完成了这项工作,通过迭代所有调用我的方法的方法来检查使用CallGraph.CallersFor(),以查看是否有任何父方法具有该属性。

这适用于检查与要检查的方法相同的程序集中的父方法,但是在线阅读时,似乎有一次CallGraph.CallersFor() 确实查看了所有程序集,但现在没有。

问题:有没有办法获取调用给定方法的方法列表,包括不同程序集中的方法?

替代答案:如果上述方法不可行,我如何遍历给定方法调用的每个方法,包括不同程序集中的方法。


示例:

-----In Assembly A

public class ClassA

    public MethodA()
    
        MethodB();
    

    public MethodB()
    
        object o = new object(); // Allocation i want to break the rule
        // Currently my rule walks up the call tree,
        // checking for a calling method with the NoAllocationsAllowed attribute.
        // Problem is, because of the different assemblies,
        // it can't go from ClassA.MethodA to ClassB.MethodB.
    



----In Assembly B

public var ClassAInstance = new ClassA();

public class ClassB

    [NoAllocationsAllowed] // Attribute that kicks off the rule-checking.
    public MethodA()
    
        MethodB();
    

    public MethodB()
    
        ClassAInstance.MethodA();
    

我真的不介意规则在哪里报告错误,在这个阶段得到错误就足够了。

【问题讨论】:

我不确定我是否真的了解您要验证的内容。您能否提供一个应该生成规则违规的代码的具体示例? 【参考方案1】:

我通过在我的 FxCop 项目中添加所有引用的 dll 解决了这个问题,并使用下面的代码手动构建了一个调用树(它还添加了对派生类的调用以解决我遇到的另一个问题,here。

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;
    

【讨论】:

【参考方案2】:

这种方式你试过了吗,

    StackTrace stackTrace = new StackTrace();
    MethodBase methodBase = stackTrace.GetFrame(1).GetMethod();
    object [] items = methodBase.GetCustomAttributes(typeof (NoAllocationsAllowed));
    if(items.Length > 0)
        //do whatever you want! 

【讨论】:

感谢您的回答,但我不想在运行时或通过反射来执行此操作。我正在尝试使用 FxCop 执行此规则,该规则执行静态检查,而不是运行时检查。

以上是关于确定一个方法是不是调用另一个包含新语句的程序集中的方法,反之亦然的主要内容,如果未能解决你的问题,请参考以下文章

在sql语句中怎么判断一个字段是不是包含在另一个字符串中

在 JavaScript/CoffeeScript 中确定一个数组是不是包含另一个数组的内容

一个方法中的hashMap的值,怎么在另一个类中调用啊?

jQuery - 确定一个分号分隔的字符串是不是包含另一个字符串

如果一个同步方法调用另一个非同步方法,非同步方法是不是有锁

JAVA中如何判断一个字符串是不是为另一个字符串的子串