gpt4 book ai didi

c# - 使用 Roslyn 替换方法

转载 作者:太空宇宙 更新时间:2023-11-03 13:37:40 27 4
gpt4 key购买 nike

我正在尝试使用 Roslyn 合并两个 C# 文件内容。我需要用新版本替换旧版本中的某些方法(标记为更新)。我已经遍历了两个文档中的所有成员,并将需要更改的方法标记到列表中。

class Token
{
public string Name; // unique identifier
public SyntaxNode Node; // actual node in the document
}

现在我有两个 token 列表,即 OldTokens 和 NewTokens。我需要用相应的新方法覆盖旧方法。

foreach (Token oldt in OldTokens)
{
Token newt = NewTokens.Find(m => m.Name == oldt.Name);
if (newt != null)
{
DocumentRoot = DocumentRoot.ReplaceNode(oldt.Node, newt.Node);
}
}

它不工作。我认为这可能是因为 DocumentRoot 没有将旧节点保存在它的直接成员列表中,因为它会出现在命名空间->类->成员的层次结构的深处。所以我尝试了类似的方法:

var nd = existing.Node.Parent.ReplaceNode(existing.Node, miss.Node);
var d2 = existing.Node.Parent.Parent.ReplaceNode(existing.Node.Parent, nd);
DocumentRoot = existing.Node.Parent.Parent.Parent.ReplaceNode(existing.Node.Parent.Parent, d2) as CompilationUnitSyntax;

这是可行的,但太难处理了。当存在嵌套类时,它会变得一团糟。有没有更好的办法?

其次,在结果中,我遗漏了代码注释。我尝试使用 ReplaceNode 方法的重载,这很令人困惑,因为它不会采用新方法节点。

谢谢

最佳答案

由于 Roslyn SyntaxNode 对象是不可变的,每次调用“ReplaceNode”时,您都会得到一棵全新的树。解决这个问题的正常方法是为每个要替换的节点添加一个SyntaxAnnotation。这是一个稳定的标记,只要节点本身未被替换,就会通过树转换保留下来。

或者,当像这样对文件进行多次更改时,您可以考虑实现 SyntaxRewriter,而不是多次调用 ReplaceNodeSyntaxRewriter 将自下而上地访问树,因此传递给该方法的节点将始终来自原始 SyntaxTree。但是,请确保您首先在覆盖的任何方法中调用 base.Visit() 并使用结果,以便您拥有作为当前节点后代的任何节点的更新版本。

最后,关于注释,您需要从要替换的节点复制琐事 (GetLeadingTrivia()/GetTrailingTrivia()) 以保留它.

关于c# - 使用 Roslyn 替换方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18208340/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com