繁体   English   中英

使用Roslyn更改语法标记

[英]Changing syntax token with Roslyn

我正在尝试用Roslyn重写代码。 我想将GreaterThanToken更改为EqualsEqualsToken。 到目前为止,这是我的代码:

ToParse.cs:

public class ToParse
{
    public bool MethodToConvert(int param)
    {
        return (1 > param);
    }
}

Program.cs中:

class Rewriter : SyntaxRewriter
{
    private readonly SyntaxKind _replace;
    private readonly SyntaxKind _replacewith;
    public Rewriter(SyntaxKind replace, SyntaxKind replacewith)
    {
        _replace = replace;
        _replacewith = replacewith;
    }

    public override SyntaxToken VisitToken(SyntaxToken token)
    {
        if (token.Kind != _replace)
            return token;

        return Syntax.Token(_replacewith);
    }
}

用法:

var code = new StreamReader("ToParse.cs").ReadToEnd();
var tree = SyntaxTree.ParseText(code);
var root = tree.GetRoot();
var rewriter = new Rewriter(SyntaxKind.GreaterThanToken, SyntaxKind.EqualsEqualsToken);
var newRoot = rewriter.Visit(root);

var newTree = SyntaxTree.Create((CompilationUnitSyntax)newRoot);
var compilation = Compilation.Create("TestAssembly.dll", 
      new CompilationOptions(OutputKind.DynamicallyLinkedLibrary),
      references: new[]{ new MetadataFileReference(typeof(object).Assembly.Location)},
      syntaxTrees: new[] { newTree });
Console.WriteLine(newTree);
EmitResult res;

using (var file = new FileStream("e:\\TestAssembly.dll", FileMode.Create))
   res = compilation.Emit(file);

执行后,Console.WriteLine打印更改的令牌return (1 == param); 但是当我用ilspy打开testassembly.dll时,我仍然看到return 1 > param; 有什么建议?

[注意:你使用的是较旧版本的Roslyn。 这个答案也适用于该版本,但我可以通过更新的名称引用类和成员,以便它们与CodePlex上可用的源匹配。

您已解析的原始树包含BinaryExpressionSyntax节点,其SyntaxKindGreaterThanExpression 当你换出GreaterThanTokenEqualsEqualsToken这里面BinaryExpressionSyntax ,它不会自动调整包含SyntaxNode实物 EqualsExpression

因此,您最终会得到一个带有EqualsEqualsTokenGreaterThanExpression 由于这不是编译器本身可以合法生成的语法树,因此您可能会看到类似的意外行为。

要在这种情况下生成正确的树,我建议通过重写CSharpSyntaxRewriter.VisitBinaryExpression并执行以下操作来重写节点本身而不是令牌:

public override SyntaxNode VisitBinaryExpression(BinaryExpressionSyntax node)
{
    if (node.CSharpKind() == SyntaxKind.GreaterThanExpression)
    {
        return SyntaxFactory.BinaryExpression(SyntaxKind.EqualsExpression, node.Left, node.Right);
    }

    return node;
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM