我正在使用Roslyn更改代码,但是,更改后生成了新的SyntaxNode
,但是我找不到执行此新代码的方法。我发现的唯一一个方法是从新的ToString
获取Root
并使用新的字符串调用EvaluateAsync
。应该有一种方法可以胜过这个,因为我已经编译了一个新代码。
static void Main(string[] args)
{
var expression = "System.Console.WriteLine(\"Test\")";
var compile = CSharpScript.Create<EntityRepresentation>(expression).GetCompilation();
var root = compile.SyntaxTrees.Single().GetRoot();
var descentands = root.DescendantNodes().Where(n =>
{
if (n is ArgumentSyntax)
return true;
return false;
}).ToList();
var otherRoot = root.ReplaceNodes(descentands, (n1, n2) =>
{
var argumentName = Argument(LiteralExpression(SyntaxKind.StringLiteralExpression, Literal("NewValue")));
return argumentName;
});
var newCode = otherRoot.ToString();
// Faz o que estou querendo, contudo não me parece a melhor maneira
var result = CSharpScript.EvaluateAsync(newCode).Result;
}
答案 0 :(得分:2)
不幸的是,从语法树创建Script对象的方法是Microsoft程序集的内部。
但是,您不必编译两次-您可以在第一次分析后再进行第二次编译。
var expression = "System.Console.WriteLine(\"Test\")";
var origTree = CSharpSyntaxTree.ParseText(expression,
CSharpParseOptions.Default.WithKind(SourceCodeKind.Script));
var root = origTree.GetRoot();
// -Snip- tree manipulation
var script = CSharpScript.Create(otherRoot.ToString());
var errors = script.Compile();
if(errors.Any(x => x.Severity == DiagnosticSeverity.Error)) {
throw new Exception($"Compilation errors:\n{string.Join("\n", errors.Select(x => x.GetMessage()))}");
}
await script.RunAsync();