10

I have stumbled upon one issue with interpolated strings for a several times now.

Consider the following case:

double number = 123.4567;
var str = $"{{{number:F2}}}"; //I want to get "{123.45}"
Console.WriteLine(str); // Will print "{F2}"

A little surprising at first but once you realize how the curly brackets are paired it makes sense. Two following curly brackets are an escape sequence for a single curly in the interpolated string. So the opening bracket of the interpolated expression is paired with the last curly in the string.

     ___pair____
    |           |
$"{{{number:F2}}}";

Now you could do the following to break the escape sequence:

var str = $"{{{number:F2} }}"; // This will be "{123.45 }"

Notice the space character this approach adds to the output. (Not ideal)

My question:

Lets say I want to use a single interpolated string to get exactly the output "{123.45}"

Is this at all possible without doing something hackish like the following?

var s = $"{{{number:F2}{'}'}";
8
  • blogs.msdn.microsoft.com/brada/2004/01/14/string-formatting-faq Commented May 28, 2018 at 18:01
  • 1
    To get the desired output, you can cheat with ToString: $"{{{number.ToString("F2")}}}" Commented May 28, 2018 at 18:05
  • @Xiaoy312 Probably the best solution yet. I have to say, that it is kinda unintuitive, that it resolved the parity of brackets differently in this case. Commented May 28, 2018 at 18:20
  • How do you feel about $"{{{$"{number:F2}"}}}"? I guess technically it's two strings, but at least it doesn't break the parser. Same idea as @Xiaoy312, format separately and insert into the string. It's the formatting that has the odd parsing rules, similar to the workarounds you have to do if you want to use the conditional (?:) operator, the existence of the colon causes issues. Commented May 28, 2018 at 18:32
  • Perhaps another less hackish alternative would just be to break the string. $"{{{number:F2}"+"}", Commented May 28, 2018 at 18:50

2 Answers 2

4

This is an expected behavior of string interpolation. It is mentioned at this Microsoft document. The below content is from Microsoft link only.

Opening and closing braces are interpreted as starting and ending a format item. Consequently, you must use an escape sequence to display a literal opening brace or closing brace. Specify two opening braces ("{{") in the fixed text to display one opening brace ("{"), or two closing braces ("}}") to display one closing brace ("}"). Braces in a format item are interpreted sequentially in the order they are encountered. Interpreting nested braces is not supported.

The way escaped braces are interpreted can lead to unexpected results. For example, consider the format item "{{{0:D}}}", which is intended to display an opening brace, a numeric value formatted as a decimal number, and a closing brace. However, the format item is actually interpreted in the following manner:

  • The first two opening braces ("{{") are escaped and yield one opening brace.
  • The next three characters ("{0:") are interpreted as the start of a format item.
  • The next character ("D") would be interpreted as the Decimal standard numeric format specifier, but the next two escaped braces ("}}") yield a single brace. Because the resulting string ("D}") is not a standard numeric format specifier, the resulting string is interpreted as a custom format string that means display the literal string "D}".
  • The last brace ("}") is interpreted as the end of the format item.
  • The final result that is displayed is the literal string, "{D}". The numeric value that was to be formatted is not displayed.

One way to write your code to avoid misinterpreting escaped braces and format items is to format the braces and format item separately. That is, in the first format operation display a literal opening brace, in the next operation display the result of the format item, then in the final operation display a literal closing brace. The following example illustrates this approach.

int value = 6324;
string output = string.Format("{0}{1:D}{2}", 
                             "{", value, "}");
Console.WriteLine(output);
// The example displays the following output:
//       {6324}  
Sign up to request clarification or add additional context in comments.

Comments

2

Assuming that it is not required to use a named format string, you can use:

var s = $"{{{number:#.#0}}}";

Comments

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.