1

我有以下代码,它构建了一个调用 Roslyn 代码的 Roslyn 语句,但我遇到了字符串转义的问题。

这是代码:

        var parseStatementArgument = "var statement = Syntax.ParseStatement(\\\"Console.WriteLine (\\\"Hello {0}\\\", parameter1);\\\");";

        var st = Syntax.InvocationExpression(
                                    Syntax.MemberAccessExpression(SyntaxKind.MemberAccessExpression, Syntax.IdentifierName("Syntax"), Syntax.IdentifierName("ParseStatement")))
                                        .AddArgumentListArguments(
                                            Syntax.Argument(Syntax.LiteralExpression(
                                                SyntaxKind.StringLiteralExpression,
                                                Syntax.Literal(
                                                    text: "\"" + parseStatementArgument + "\"",
                                                    value: parseStatementArgument)
                                    )));

        var variableDeclarator = Syntax.VariableDeclarator(Syntax.Identifier("statement"))
            .WithInitializer(Syntax.EqualsValueClause(st));

        var varStatement = Syntax.VariableDeclaration(Syntax.IdentifierName("var"), Syntax.SeparatedList(variableDeclarator));

        var varStatementText = varStatement.Format().GetFormattedRoot().GetFullText() + ";";

        var scriptEngine = new ScriptEngine(
            new [] {
                MetadataReference.Create("Roslyn.Compilers"),
                MetadataReference.Create("Roslyn.Compilers.CSharp"),
                MetadataReference.Create("Roslyn.Services"),
                MetadataReference.Create("Roslyn.Services.CSharp")
            },
            new [] {
                "System",
                "Roslyn.Compilers.CSharp",
                "Roslyn.Scripting",
                "Roslyn.Scripting.CSharp",
                "Roslyn.Services"
            });

        var session = Session.Create();

        scriptEngine.Execute(varStatementText, session);

        scriptEngine.Execute("Console.WriteLine (statement.Format().GetFormattedRoot().GetFullText());", session);

问题是通过脚本引擎执行打印到控制台窗口的“语句”将错过“Hello {0}”字符串周围的反斜杠。如果我在参数中添加双重转义(附加 \ ,Roslyn 将引发有关缺少逗号的编译错误。

如何更新此代码以获取我想要的语句变量的语法正确版本?

4

2 回答 2

1

如何切换到使用逐字字符串级别,并在添加节点时添加另一个级别的转义。

就像是:

    var parseStatementArgument = @"var statement = Syntax.ParseStatement(@""Console.WriteLine (""""Hello {0}"""", parameter1);"");";
    var st = Syntax.InvocationExpression(
                                Syntax.MemberAccessExpression(SyntaxKind.MemberAccessExpression, Syntax.IdentifierName("Syntax"), Syntax.IdentifierName("ParseStatement")))
                                    .AddArgumentListArguments(
                                        Syntax.Argument(Syntax.LiteralExpression(
                                            SyntaxKind.StringLiteralExpression,
                                            Syntax.Literal(
                                                text: "@\"" + parseStatementArgument.Replace("\"", "\"\"") + "\"",
                                                value: parseStatementArgument)
                                )));
于 2012-07-11T15:49:32.457 回答
0

基于凯文关于如何替换字符串的提示,我玩过并发现这是一个可行的解决方案,但它引发了另一个问题。

解决方案:

        var parseStatementArgument = "var statement = Syntax.ParseStatement(\\\"Console.WriteLine (\\\\\\\"Hello {0}\\\\\\\", parameter1);\\\");";

        var st = Syntax.InvocationExpression(
                                    Syntax.MemberAccessExpression(SyntaxKind.MemberAccessExpression, Syntax.IdentifierName("Syntax"), Syntax.IdentifierName("ParseStatement")))
                                        .AddArgumentListArguments(
                                            Syntax.Argument(Syntax.LiteralExpression(
                                                SyntaxKind.StringLiteralExpression,
                                                Syntax.Literal(
                                                    text: "\"" + parseStatementArgument + "\"",
                                                    value: parseStatementArgument.Replace ("\\\\\\", "\\"))
                                    )));

现在它正确地输出了一个语法正确且编译良好的代码片段。

它提出的问题是我必须修改源字符串而不是派生字符串才能获得正确的结果。使用 Roslyn 重写代码或生成代码时,不需要双倍或三倍转义字符串文字以使 Roslyn 能够正确处理,也许这是 Roslyn 问题,我希望有人能阐明一个优雅的解决方案适用于所有类型的字符串。

于 2012-07-11T17:15:29.147 回答