C#如何在Roslyn中检测EvenHandler订阅

时间:2019-01-23 10:19:59

标签: c# roslyn roslyn-code-analysis roslynator

我正在使用Roslyn编写自定义分析器规则。

我想找到一个对某些事件(通过订阅)进行处理的方法。 像这样:

public class Counter
{
    public event EventHandler ThresholdReached;
}

public class TestEvent
{
    public TestEvent()
    {
        Counter с = new Counter();
        с.ThresholdReached += OnThresholdReached;
    }

    private void OnThresholdReached(object sender, EventArgs e)
    {

    }
}

在我看来,它看起来是:

    private static void HandleMethodDeclaration(SyntaxNodeAnalysisContext context)
    {
        MethodDeclarationSyntax methodDeclaration = (MethodDeclarationSyntax)context.Node;
        if (methodDeclaration.Identifier.IsMissing)
        {
            return;
        }

        IMethodSymbol methodSymbol = context.SemanticModel.GetDeclaredSymbol(methodDeclaration);

    }

我不知道如何检测 OnThresholdReached 是事件 ThresholdReached 的订阅。如果有人知道该怎么做,请帮助=)

1 个答案:

答案 0 :(得分:0)

在分析器中,仅通过查看MethodDeclarationSyntax就无法知道该方法是否转换为委托。因此,您无法(仅通过查看MethodDeclarationSyntax)就无法知道该委托是否传递给事件的添加访问器。

首先,请记住Roslyn分析器只能看到当前程序集(项目)中的用法。如果将您的方法转换为另一个程序集中的委托,则分析器将无法查看此结果。

第二,请记住

с.ThresholdReached += OnThresholdReached;

可以表示为

EventHandler handler = OnThresholdReached;
с.ThresholdReached += handler;

如果只想检测第一种情况,可以查看AssignmentExpressionSyntax类型的SyntaxKind.AddAssignmentExpression个实例,并进行分析。

如果要检测将方法组转换为委托的所有情况,则需要查看类型为SimpleNameSyntax的所有实例,并进行如下分析:

void Analyze(SyntaxNodeAnalysisContext context)
{
    var node = context.Node as SimpleNameSyntax;

    // we're only interested in delegates
    var type = context.SemanticModel.GetTypeInfo(node, context.CancellationToken).ConvertedType;

    if (type == null || type.TypeKind != TypeKind.Delegate)
    {
        return;
    }

    // we're only interested in methods from the current assembly
    var symbol = context.SemanticModel.GetSymbolInfo(node, context.CancellationToken).Symbol;

    if (symbol == null ||
        symbol.Kind != SymbolKind.Method ||
        !symbol.ContainingAssembly.Equals(context.SemanticModel.Compilation.Assembly))
    {
        return;
    }

    // now you know symbol is a method in the same assembly, that is converted to a delegate
}

要查找该方法的源代码,请参见https://stackoverflow.com/a/45362532/1403794