删除节点时保持结构化的琐事
本文关键字:结构化 节点 删除 | 更新日期: 2023-09-27 18:00:15
有没有一种简单的方法可以从树中删除SyntaxNode(即方法),但保留结构化的琐事?
在下面的代码中,我想删除方法A:
public class Sample
{
#region SomeRegion
public void MethodA()
{
}
#endregion
}
我使用CSharpSyntaxRewriter来重写SyntaxTree。在VisitMethodDeclaration方法中,我只是为MethodA返回null。这种方法的问题在于,#region标签的StructuredTrivia也被删除了。这是结果:
public class Sample
{
#endregion
}
在我的CSharpSyntaxRewriter:
public override SyntaxNode VisitMethodDeclaration(MethodDeclarationSyntax node)
{
if (...)
return null;
else
return node;
}
编辑:正如下面的一个答案中所提到的,我可以将SyntaxNode.RemoveNodes与SyntaxRemoveOptions.KeepDirectives选项一起使用。此解决方案有两大缺点:
- 我需要知道哪些SyntaxNode类型可以包含此子类型。例如,一个方法可以在Struct、Class、Interface等中声明。。。这意味着我需要在多个位置进行过滤
- 我失去了从下到上构建语法树的能力。这在比较SyntaxTree对象时会出现问题。所有对访问者方法的后续调用都已经看到了在"RemoveNodes"方法中创建的新节点。使用SyntaxNode.RemoveNodes方法可以指定SyntaxRemoveOptions.KeepDirective,但CSharpSyntaxRewriter也可以这样做吗
第二版:以下是我要做的一些代码:https://dotnetfiddle.net/1Cg6UZ
当删除节点时,实际上是在删除它的琐事。要保留琐事,需要修改ClassDeclarationSyntax而不是MethodDeclaration。
访问ClassDeclarationSyntax时,您可以通过删除适当的节点来修改类,并使用SyntaxRemoveOptions.KeepTrailingTrivia|SyntaxremovOptions.Keep LeadingTrivia在实际方法定义前后保留注释和区域语句。
public class ClassDeclarationChanger : CSharpSyntaxRewriter
{
public override SyntaxNode VisitClassDeclaration(ClassDeclarationSyntax node)
{
var methods = node.Members.OfType<MethodDeclarationSyntax>();
if (methods.Any())
{
node = node.RemoveNodes(methods, SyntaxRemoveOptions.KeepTrailingTrivia |
SyntaxRemoveOptions.KeepLeadingTrivia);
}
return base.VisitClassDeclaration(node);
}
}
如果您希望首先访问子节点,当然也可以执行base。首先访问ClassDeclaration(节点),然后仅在战后删除方法节点。
另一种方法是返回另一份声明。但是,您不能简单地返回EmptyStatement(因为这将导致异常),而是可以插入一个没有内容的新方法声明:
public class SampleChanger : CSharpSyntaxRewriter
{
public override SyntaxNode VisitMethodDeclaration(MethodDeclarationSyntax node)
{
// Generates a node containing only parenthesis
// with no identifier, no return type and no parameters
var newNode = SyntaxFactory.MethodDeclaration(SyntaxFactory.IdentifierName(""), "");
// Removes the parenthesis from the Parameter List
// and replaces them with MissingTokens
newNode = newNode.ReplaceNode(newNode.ParameterList,
newNode.ParameterList.WithOpenParenToken(
SyntaxFactory.MissingToken(SyntaxKind.OpenParenToken)).
WithCloseParenToken(SyntaxFactory.MissingToken(SyntaxKind.CloseParenToken)));
// Returns the new method containing no content
// but the Leading and Trailing trivia of the previous node
return newNode.WithLeadingTrivia(node.GetLeadingTrivia()).
WithTrailingTrivia(node.GetTrailingTrivia());
}
}
当然,这种方法的缺点是,它需要针对不同语法类型的特定SyntaxNode,因此在代码的其他部分重用它可能很困难。
我想你正在寻找标志KeepExteriorTrivia
如果你检查枚举的源,你会发现他们有一个为预先编译的标志
[Flags]
public enum SyntaxRemoveOptions
{
KeepNoTrivia = 0,
KeepLeadingTrivia = 1,
KeepTrailingTrivia = 2,
KeepExteriorTrivia = KeepTrailingTrivia | KeepLeadingTrivia,
KeepUnbalancedDirectives = 4,
KeepDirectives = 8,
KeepEndOfLine = 16,
AddElasticMarker = 32,
}
你的函数调用现在看起来是这样的:
public class ClassDeclarationChanger : CSharpSyntaxRewriter
{
public override SyntaxNode VisitClassDeclaration(ClassDeclarationSyntax node)
{
var methods = node.Members.OfType<MethodDeclarationSyntax>();
if (methods.Any())
{
node = node.RemoveNodes(methods, SyntaxRemoveOptions.KeepExteriorTrivia);
}
return base.VisitClassDeclaration(node);
}
}