. NET String.Format Включить счетчик со знаком минус для заполнения 0 - PullRequest
1 голос
/ 14 марта 2020

Я портирую некоторый код C ++, который использует _vsnwprintf_s для преобразования чисел в 11-символьные строки. Если число отрицательное, оно по-прежнему возвращает 11-символьную строку.

Мне нужно повторить это поведение в C#. Если я использую:

String.Format("{0:D11}",-30800322);
String.Format("{0:D11}",30800322);

Проблема в том, что отрицательное число приводит к выводу 12 символов, а не 11. Т.е. он выдает -00030800322, но я хочу, чтобы он вывел -0030800322, а 30800322 должен вывести 00030800322.

Я могу решить эту проблему, проверив, является ли число отрицательным, и используя {0:D10} для отрицательных чисел и {0:D11} для положительных чисел, однако мне интересно, есть ли лучший способ более точно воспроизвести версию C ++, которая включает знак минус на счету.

1 Ответ

3 голосов
/ 14 марта 2020

У вас есть много способов получить этот вывод. Таким образом, у вас есть две переменные для положительных и отрицательных значений:

var s = -30800322;
var u = 30800322;

1. string.Replace


Быстрый и простой способ:

Console.WriteLine(string.Format("{0:D11}", s).Replace("-0", "-"));
Console.WriteLine(string.Format("{0:D11}", u).Replace("-0", "-"));

2. IFormatProvider


Создание пользовательского IFormatProvider для использования с функцией string.Format(..):

public class vsnwprintf : IFormatProvider, ICustomFormatter
{
    public string Format(string format, object arg, IFormatProvider formatProvider)
    {
        if (arg == null || !int.TryParse(arg.ToString(), out int x))
            return null;

        var f = "{0:D11}";

        if (x < 0)
            return string.Concat("-", string.Format(f, x).Substring(2));
        else
            return string.Format(f, x);
    }

    public object GetFormat(Type formatType)
    {
        return (formatType == typeof(ICustomFormatter)) ? this : null;
    }
}

Использование:

Console.WriteLine(string.Format(new vsnwprintf(), "{0}", s));
Console.WriteLine(string.Format(new vsnwprintf(), "{0}", u));

3. ToString ()


Создать новый класс и переопределить функцию ToString():

public class vsnwprintf
{
    private readonly int Value;
    private readonly int Length;

    private vsnwprintf() { }

    public vsnwprintf(int val, int length) : this()
    {
        Value = val;
        Length = length;
    }

    public override string ToString()
    {
        var f = string.Concat("{0:D", Length, "}");

        if (Value < 0)
            return string.Concat("-", string.Format(f, Value).Substring(2));
        else
            return string.Format(f, Value);
    }
}

Использование:

Console.WriteLine(new vsnwprintf(s, 11));
Console.WriteLine(new vsnwprintf(u, 11));

4. Метод расширения


Создание класса stati c и метод расширения для типа int:

public static class Extensions
{
    public static string Tovsnwprintf(this int Value, int Length = 11)
    {
        var f = string.Concat("{0:D", Length, "}");

        if (Value < 0)
            return string.Concat("-", string.Format(f, Value).Substring(2));
        else
            return string.Format(f, Value);
    }
}

Использование:

Console.WriteLine(s.Tovsnwprintf());
Console.WriteLine(u.Tovsnwprintf(11));

Каждый из которых возвращает:

-0030800322
00030800322
...