How to Calculate Double + Float Precision

后端 未结 4 2168
挽巷
挽巷 2020-12-03 08:30

I have been trying to find how to calculate the Floating/Double precision/range numbers -3.402823e38 .. 3.402823e38 and -1.79769313486232e308 .. 1.79769313486232e308.

相关标签:
4条回答
  • 2020-12-03 08:57

    it's not that easy to calculate. this is because of the way that floats and doubles are implemented. they are both devided in two parts: one part for the base number, and one part for the exponent. i think float is divided 24 bits to base number and 8 bits to exponent. but i'm not sure about this! i'll base my further calculations and assumptions on this fact, though, so these calculations may all be wrong, but they illustrate the correct principles. these specifications could also differ from language to language, even though there are standards which say they shouldn't. but nothing can be taken for granted while programming :p

    this means that the base number can be between -8388608 and 8388607, and the exponent can be between -128 and 127.

    then when the number is used, the computer thinks like this:

    base * 10^exponent

    which leads to the biggest possible number being 8388607 * 10 ^ 127, which would be... a lot. but it would contain 120 zeros, cause it can't specify more than 7 more numbers.

    the accuracy of the number gets lower as the number grows. this means that your question is badly formatted ;) you can only specify a valid range for a float, if you know what number of correct decimals are needed. if you need the possibility of 2 guaranteed accurate decimals, then float would have a range of -83885 to 83885.

    0 讨论(0)
  • 2020-12-03 09:03

    I've discussed the floating point format from various (though not all of course) aspects. This answer may assist you: https://stackoverflow.com/questions/4851671/printing-double-without-losing-precision

    0 讨论(0)
  • 2020-12-03 09:04

    The float type can represent values ranging from approximately 1.5 × 10−45 to 3.4 × 1038 with a precision of 7 digits.

    The double type can represent values ranging from approximately 5.0 × 10−324 to 1.7 × 10308 with a precision of 15–16 digits.

    http://msdn.microsoft.com/en-us/library/aa691146%28v=vs.71%29.aspx

    0 讨论(0)
  • 2020-12-03 09:16

    Well, both types actually look like the following:

    [sign] [exponent] [mantissa]
    

    representing a number in the following form:

    [sign] 1.[mantissa] × 2[exponent]

    with the size of the exponent and mantissa varying. For float the exponent is eight bits wide, while double has an eleven-bit exponent. Furthermore, the exponent is stored unsigned with a bias which is 127 for float and 1023 for double. This results in a range for the exponent of −126 through 127 for float and −1022 though 1023 for double.

    The exponent is the exponent for 2something so when calculating 2127 you'll get 1.7 × 1038 which gets you in the approximate range of the float maximum value. Similarly for double with 9 × 10307.

    Obviously those numbers are not exactly those we expect. This is where the mantissa comes into play. The mantissa represents a normalized binary number that always begins with “1.” (that's the normalized part). The rest is simply the digits after the dot. Since the maximum mantissa is then roughly 1.111111111... in binary, which is almost 2, we'll get approximately 3.4 × 1038 as float's maximum value and 1.79 × 10308 as the maximum value for double.

    [EDIT 2011-01-06] As Mark points out below (and below the question), the exact formula is the following:

    Formula to calculate the exact maximum value for an IEEE-754 floating-point type: 2^(2^(e-1) )⋅(1-2^(-p) )

    where e is the number of bits in the exponent and p is the number of bits in the mantissa, including the aforementioned implicit bit (due to normalization). The formula replicates what we have seen above, only now accurate. The first factor, 22e − 1, is the maximum exponent, multiplied by two (we save the two in the second factor then). The second factor is the largest number we can represent below one. I said above that the number is almost two. Since we exaggerated the exponent by a factor of two in this formula, we need to account for that and now have a number that is almost one. I hope it's not too confusing.

    In any case, for float (with e = 8 and p = 24) we get the exact value 340282346638528859811704183484516925440 or roughly 3.4 × 1038. double then yields (with e = 10 and p = 53) 179769313486231570814527423731704356798070567525844996598917476803157260780028538760589558632766878171540458953514382464234321326889464182768467546703537516986049910576551282076245490090389328944075868508455133942304583236903222948165808559332123348274797826204144723168738177180919299881250404026184124858368 or roughly 1.80 × 10308.

    [/EDIT]

    Another thing: You're bringing up the term “precision” in your question but you quote the ranges of the types. Precision is a quite different thing and refers to how many significant digits the type can retain. Again, the answer here lies in the mantissa which is 23 and 52 bits for float and double, respectively. Since the numbers are stored normalized we actually have an implicit bit added to that, which puts us at 24 and 53 bits. Now, the way how digits after the decimal (or binary here) point work is the following:

     1.   1     0     1     1
     ↑    ↑     ↑     ↑     ↑
    2^0  2^-1  2^-2  2^-3  2^-4
     =    =     =     =     =
     1   0.5   0.25  0.125 0.0625
    

    So the very last digit in the double mantissa represents a value of roughly 2.2 × 10−16 or 2−52, so if the exponent is 1, this is the smallest value we can add to the number – placing the double precision around 16 decimal digits. Likewise for float with roughly seven digits.

    0 讨论(0)
提交回复
热议问题