Преобразование double в hex в C#

У меня есть это значение:

double headingAngle = 135.34375;

Я хотел бы преобразовать его в HEX и распечатать HEX на консоли. Я уже преобразовал строку и int в их соответствующие шестнадцатеричные значения, но double кажется гораздо более сложным. Кто-нибудь может указать мне правильное направление?

5 ответов


Ну, я гуглил минуту или две и в соответствии с этой вот довольно эллегантное решение

    double d = 12.09;
    Console.WriteLine("Double value: " + d.ToString());
    byte[] bytes = BitConverter.GetBytes(d);
    Console.WriteLine("Byte array value:");
    Console.WriteLine(BitConverter.ToString(bytes));

вы можете преобразовать базу 10 в базу 16, постоянно умножая дробь на 16, вычеркивая "целое" число и повторяя с остатком.

Итак, чтобы преобразовать 0,1 десятичного в шестнадцатеричный

0.1 * 16
= 1.6

так 1 становится первым шестнадцатиричное значение. Продолжай оставшиеся 0,6

0.6 * 16 = 9.6

так 9 становится вторым шестнадцатеричное значение. Продолжая идти с оставшимися 0,6

0.6 * 16 = 9.6

etc.

так 0.1 Decimal = 0.19999.. повторяющийся hex

из памяти это работает для любого radix. Очевидно, в hex целое значение 10 будет etc.


предполагая, что вы хотите преобразовать в шестнадцатеричную базу / радиус, следующее должно сделать трюк:

static void Main(string[] args)
{
    Console.WriteLine(Base16(135.34375, 10));
    Console.ReadLine();
}

private static string Base16(double number, int fractionalDigits)
{
    return Base(number, fractionalDigits, new char[]{
        '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
        'A', 'B', 'C', 'D', 'E', 'F' });
}

private static string Base(double number, int fractionalDigits, params char[] characters)
{
    int radix = characters.Length;
    StringBuilder sb = new StringBuilder();

    // The 'whole' part of the number.
    long whole = (long)Math.Floor(number);
    while (whole > 1)
    {
        sb.Insert(0, characters[whole % radix]);
        whole = whole / radix;
    }

    // The fractional part of the number.
    double remainder = number % 1;
    if (remainder > Double.Epsilon || remainder < -Double.Epsilon)
    {
        sb.Append('.');

        double nv;
        for (int i = 0; i < fractionalDigits; i++)
        {
            nv = remainder * radix;
            if (remainder < Double.Epsilon && remainder > -Double.Epsilon)
                break;
            sb.Append(characters[(int)Math.Floor(nv)]);
            remainder = nv % 1;
        }
    }

    return sb.ToString();
}

шестнадцатеричное преобразование 135.34375 является 87.58.


попробуйте это:

public static string Format(double number, double @base)
{
    StringBuilder format = new StringBuilder();
    if(number < 0)
    {
        format.Append('-');
        number = -number;
    }
    double log = Math.Log(number, @base);
    bool frac = false;
    double order;
    if(log < 0)
    {
        frac = true;
        format.Append(digits[0]);
        format.Append('.');
        order = 1/@base;
    }else{
        order = Math.Pow(@base, Math.Floor(log));
    }
    while(number != 0 || order >= 1)
    {
        double digit = Math.Floor(number/order);
        if(digit >= @base) break;
        number = number-digit*order;
        number = Math.Round(number, 15);
        if(order < 1 && !frac)
        {
            format.Append('.');
            frac = true;
        }
        order /= @base;
        if(digit >= 10)
        {
            format.Append((char)('A'+(digit-10)));
        }else{
            format.Append((char)('0'+digit));
        }
    }
    return format.ToString();
}

использовать его как Format(headingAngle, 16). Вы также можете прокомментировать number = Math.Round(number, 15); для более интересных результатов. ☺

результат находится в инвариантном к культуре формате. Для 135.34375, он возвращает 87.58.


BitConverter.DoubleToInt64Bits(value).ToString("X")