将“float”转换为“byte[4]”,再在.NET Micro Framework中将其转换回“float”。

6

如何将 float 转换为 byte[4],然后再转回 'float' 呢?

我在使用 C# .NET 微框架 进行操作,所以无法使用 BitConverter


1
小心字节序(@Mitch的链接显示了一个使用小端序的BitConverter类) - user568493
1
编码浮点数的平台没有任何类型的BitConverter类。任何使用BitConverter类的代码都将无法工作。 - chris12892
没关系,看起来它是开源的,我可以提取出我需要的部分。 - chris12892
我假设你已经尝试了之前提到的链接中的类(据说是为C# .NETMF编写的)。请解释一下它为什么不能工作:你尝试了哪些函数?期望输出是什么? - user568493
我无法使用它,因为这些字节在两个目标之间进行交换。其中一个目标没有运行NETMF。它们在一个目标上被编码,在另一个目标上被解码。另一个目标是Arduino。 - chris12892
显示剩余4条评论
3个回答

11

uint 转换为 float(反之亦然)也可以使用“安全”的代码进行转换(但我不知道在NETMF上是否可能实现)。

[StructLayout(LayoutKind.Explicit)]
struct UIntFloat
{       
    [FieldOffset(0)]
    public float FloatValue;

    [FieldOffset(0)]
    public uint IntValue;        
}

public static float ToSingle(byte[] value, int index)        
{           
    uint i = ToUInt32(value, index);            
    return ToSingle(i);
}

public static float ToSingle(uint value)
{
    UIntFloat uf = new UIntFloat();
    uf.IntValue = value;
    return uf.FloatValue;
}

如果另一个变量是uint还是int,这会有任何区别吗? - R1PFake

3

我修改了Netduino实现中的BitConverter类,使得可以指定字节序(这不是“最好的方法”,但它能正常工作)。如果要将字节数组发送到网络上,我会使用BigEndian。需要提醒的是,在.NET Micro Framework中,unsafe并没有官方支持。

using System;
using System.Diagnostics;

namespace netduino
{
    public static class BitConverter
    {
        public static byte[] GetBytes(uint value)
        {
            return new byte[4] { 
                    (byte)(value & 0xFF), 
                    (byte)((value >> 8) & 0xFF), 
                    (byte)((value >> 16) & 0xFF), 
                    (byte)((value >> 24) & 0xFF) };
        }

        public static unsafe byte[] GetBytes(float value)
        {
            uint val = *((uint*)&value);
            return GetBytes(val);
        }

        public static unsafe byte[] GetBytes(float value, ByteOrder order)
        {
            byte[] bytes = GetBytes(value);
            if (order != ByteOrder.LittleEndian)
            {
                System.Array.Reverse(bytes);
            }
            return bytes;
        }

        public static uint ToUInt32(byte[] value, int index)
        {
            return (uint)(
                value[0 + index] << 0 |
                value[1 + index] << 8 |
                value[2 + index] << 16 |
                value[3 + index] << 24);
        }

        public static unsafe float ToSingle(byte[] value, int index)
        {
            uint i = ToUInt32(value, index);
            return *(((float*)&i));
        }

        public static unsafe float ToSingle(byte[] value, int index, ByteOrder order)
        {
            if (order != ByteOrder.LittleEndian)
            {
                System.Array.Reverse(value, index, value.Length);
            }
            return ToSingle(value, index);
        }

        public enum ByteOrder
        {
            LittleEndian,
            BigEndian
        }

        static public bool IsLittleEndian
        {
            get
            {
                unsafe
                {
                    int i = 1;
                    char* p = (char*)&i;

                    return (p[0] == 1);
                }
            }
        }
    }
}

namespace BitConverterTest
{
    class Program
    {
        static void Main(string[] args)
        {
            byte[] msbFirst = new byte[] { 0x42, 0xF6, 0xE9, 0xE0 };
            byte[] lsbFirst = new byte[] { 0xE0, 0xE9, 0xF6, 0x42 };
            const float f = 123.456789F;

            byte[] b = netduino.BitConverter.GetBytes(f, netduino.BitConverter.ByteOrder.BigEndian);
            for (int i = 0; i < b.Length; i++)
            {
                Debug.Assert(msbFirst[i] == b[i], "BitConverter.GetBytes(float, BigEndian) i=" + i);
            }

            Debug.Assert(f == netduino.BitConverter.ToSingle(msbFirst, 0, netduino.BitConverter.ByteOrder.BigEndian));

            Console.WriteLine("All tests passed");
            Console.ReadKey();
        }
    }
}

-1

这对我有用,可能不是最完整的答案,但很简单

void floatToByte(GLubyte b[], float n)
{
unsigned int val = *((unsigned int*)&n);

b[0] = (GLubyte)(val  & 0xFF);
b[1] = (GLubyte)((val >> 8) & 0xFF);
b[2] = (GLubyte)((val >> 16) & 0xFF);
b[3] = (GLubyte)((val >> 24) & 0xFF);   
}



float byteToFloat(GLubyte b[])
{
unsigned int ret =  (unsigned int)(b[0] << 0 | b[1] << 8 | b[2] << 16 | b[3] << 24);
float r = *(((float*)&ret));
return r;
}

3
这甚至不是C#! - GenericPtr

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接