51

FormattableString已在 C# 6.0 中引入。由于我们可以使用stringobject 使用相同的字符串格式,为什么需要使用FormattableStringor IFormattable。三者有什么区别?

我的代码

        var name = "Pravin";
        var s = $"Hello, {name}";
        System.IFormattable s1 = $"Hello, {name}";
        System.FormattableString s2 = $"Hello, {name}";

最重要的是那里产生相同的结果。IE'Hello Pravin'.

如果有人对此有深入的了解,我能否得到更详细的答案。

4

2 回答 2

71

FormattableString是 .NET 4.6 中的一种新类型,编译器只有在你尝试使用它时才会使用它。换句话说,插值字符串文字的类型通常是string- 用 - 构建的- 但如果您要求它string.Format可以是FormattableString(via )。FormattableStringFactory

AFormattableString由将传递给string.Format(例如"Hello, {0}")的格式字符串和将传递以对其进行格式化的参数组成。至关重要的是,此信息在格式化之前被捕获。

这允许您适当地调整格式 - 最常见的是在不变的文化中执行它,通常使用Invariantstatic 方法

当您将插值字符串文字分配给IFormattable变量时,它也将使用FormattableString。在这种情况下,IFormattable.ToString(string, CultureInfo)实现忽略了第一个参数,这可能是它使用显式接口实现的原因。

示例代码:

using System;
using System.Globalization;
using System.Threading;
using static System.FormattableString;

class Test
{
    static void Main()
    {
        var uk = CultureInfo.CreateSpecificCulture("en-GB");
        Thread.CurrentThread.CurrentCulture = uk;
        var germany = CultureInfo.CreateSpecificCulture("de-DE");
        string now = $"Default: it is now {DateTime.UtcNow}";
        Console.WriteLine(now); // UK format
        IFormattable x = $"Specific: It is now {DateTime.UtcNow}";
        Console.WriteLine(x.ToString("ignored", germany));
        FormattableString y = $"FormattableString: It is now {DateTime.UtcNow}";
        Console.WriteLine(FormattableString.Invariant(y));
        // Via using static
        Console.WriteLine(Invariant($"It is now {DateTime.UtcNow}")); 
    }
}

样本结果:

Default: it is now 16/02/2016 07:16:21
Specific: It is now 16.02.2016 07:16:21
FormattableString: It is now 02/16/2016 07:16:21
It is now 02/16/2016 07:16:21
于 2016-02-16T07:09:58.257 回答
6

顺便说一句,https: //www.meziantou.net/interpolated-strings-advanced-usages.htm 涵盖了一些 FormattableString 允许您执行的操作的示例(例如,自动参数化 SQL 语句)

例如

void ExecuteNonQuery(DbConnection connection, FormattableString formattableString)
{
    using (var command = connection.CreateCommand())
    {
        // Replace values by @p0, @p1, @p2, ....
        var args = Enumerable.Range(0, formattableString.ArgumentCount).Select(i => (object)("@p" + i)).ToArray();

        command.CommandType = System.Data.CommandType.Text;
        command.CommandText = string.Format(formattableString.Format, args);

        // Create parameters
        for (var i = 0; i < formattableString.ArgumentCount; i++)
        {
            var arg = formattableString.GetArgument(i);
            var p = command.CreateParameter();
            p.ParameterName = "@p" + i;
            p.Value = arg;
            command.Parameters.Add(p);
        }

        // Execute the command
        command.ExecuteNonQuery();
    }
}

using (var sqlConnection = new SqlConnection())
{
    sqlConnection.Open();
    ExecuteNonQuery(sqlConnection, $@"UPDATE Customers SET Name = {"Meziantou"} WHERE Id = {1}");
}

于 2020-04-20T18:51:19.043 回答