How to output IEEE-754 format integer as a float

2020-07-17 06:53发布

问题:

I have a unsigned long integer value which represents a float using IEEE-754 format. What is the quickest way of printing it out as a float in C++?

I know one way, but am wondering if there is a convenient utility in C++ that would be better.

Example of the way that I know is:

union
{
    unsigned long ul;
    float f;
} u;

u.ul = 1084227584; // in HEX, this is 0x40A00000

cout << "float value is: " << u.f << endl;

(This prints out "float value is: 5" )

回答1:

The union method you suggested is the usual route that most people would take. However, it's technically undefined behavior in C/C++ to read a different member from a union than the one that was most recently written. Despite this, though, it's well-supported among pretty much all compilers.

Casting pointers, as Jon Skeet suggested, is a bad idea -- that violates the strict aliasing rules of C. An aggressive optimizing compiler will produce incorrect code for this, since it assumes that pointers of types unsigned long * and float * will never alias each other.

The most correct way, in terms of standards compliance (that is, not invoking undefined behavior), is to cast through a char*, since the strict aliasing rules permit a char* pointer to alias a pointer of any other type:

unsigned long ul = 0x40A00000;
float f;
char *pul = (char *)&ul;  // ok, char* can alias any type
char *pf = (char *)&f;    // ok, char* can alias any type
memcpy(pf, pul, sizeof(float));

Though honestly, I would just go with the union method. From the cellperformance.com link above:

It is an extremely common idiom and is well-supported by all major compilers. As a practical matter, reading and writing to any member of a union, in any order, is acceptable practice.



回答2:

It will only work on 32-bit machines or machines where sizeof(long) == sizeof(float). On modern machines you might want to use int instead ... and you really have to take care if you're performing this trick.



回答3:

I was thinking the same thing as Jon Skeet, but he beat me to it. However, I would do it a little more concisely than he did.

cout << "float value is: " << *((float *) &ulValue)) << endl;

You get a pointer to your unsigned long, then reinterpret that as a pointer to float, then dereferencing your pointer to float produces the float value.

Since you are doing this in C++, it is clearer to use reinterpret_cast instead of the old C-style cast.

cout << "float value is: " << *(reinterpret_cast<float *>(&ulValue)) << endl;


回答4:

EDIT: I previously thought this was "just nasty", but it turns out to be worse than I thought - see the comments for details. I wouldn't recommend this approach, but I'm leaving it here to document the possibility (and the caveat!)


You can use pointers:

long ul = 1084227584;
float* fp = (float*) &ul;
float f = *fp;

(This can be condensed into a single line, but I think it's clearer not to.)

Basically the same thing as your union... and equally dodgy unless you're sure of the sizes of the types involved.

If your platform supports them, you should probably use the size-specific type names for both the integer and floating point types.



回答5:

swegi had the right direction, but missed one character. The correct way is

long l = 1084227584L
float f = reinterpret_cast<float&>(l);