Преобразование 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.