1

正如您可能已经猜到的那样,我正在学习有关 roslyn 的所有知识,尤其是作为代码分析器。

语法突出显示效果很好。但是,以下 - 这是我的代码操作 - 在删除节点时会以静默方式失败:

    private async Task<Document> RemoveNode(Document document, LocalDeclarationStatementSyntax typeDecl, CancellationToken cancellationToken)
    {
        IEnumerable<SyntaxNode> oldNode = typeDecl.DescendantNodes().OfType<VariableDeclarationSyntax>();

        SyntaxNode oldRoot = await document.GetSyntaxRootAsync(cancellationToken);

        SyntaxNode newRoot = oldRoot.RemoveNode(oldNode.Single(), SyntaxRemoveOptions.KeepNoTrivia); //Analyzer fails here

        return document.WithSyntaxRoot(newRoot);
    }

主题:

namespace ConsoleApplication3
{
    class Program
    {
        static void Main(string[] args)
        {
            FruitMix fm = new FruitMix(); //This is the matched node
        }
    }
}

我觉得我错过了如何与 Roslyn 合作的“更大”图景,所以虽然这里的帮助会很棒,但我也喜欢一些对我有帮助的链接/资源。

我已经在这里上传了这个项目,虽然不是一个“最小”的例子,它很容易重现这个问题。上面的代码在CodeFixProvider.cs.

谢谢

4

1 回答 1

2

当我在启用所有引发的异常的情况下运行您的代码时,我可以看到它ArgumentNullException在调用堆栈内部的深处抛出RemoveNode(),特别是在SyntaxFactory.LocalDeclarationStatement().

该异常令人困惑(已在 GitHub 上报告),但实际上是您的错误:您试图VariableDeclarationSyntax从其父级中删除LocalDeclarationStatementSyntax(其语法类似于LocalDeclarationStatement : const? VariableDeclaration ;)。而且由于LocalDeclarationStatementSyntax没有VariableDeclarationSyntax无效,因此您会遇到异常。

最简单的解决方法是删除父级LocalDeclarationStatementSyntax

SyntaxNode newRoot = oldRoot.RemoveNode(oldNode.Single().Parent, SyntaxRemoveOptions.KeepNoTrivia);
于 2016-04-08T15:44:54.253 回答