c# - 哪个 C# double 文字不能完全表示为 double?

标签 c# double literals

Wikipedia says :

For example, the decimal number 0.1 is not representable in binary floating-point of any finite precision

但是,在 C# 中

string s = 0.1.ToString(CultureInfo.InvariantCulture);
Console.WriteLine(s);

写入 0.1

我本以为会是 0.099999999999999999 左右。

我正在寻找至少一个不能完全表示为 double 的示例 double 文字。

编辑:

正如其他人指出的那样,0.1 确实是我一直在寻找的字面量,如下面的经典代码示例所示:

double sum = 0.0;

for (int i = 0; i < 10; i++)
{
    sum += 0.1;
}

Console.WriteLine(sum.Equals(1.0)); // false

当 double 转换为其他数据类型时,会发生一些奇怪的事情。这不仅是 string 的情况,因为此表达式为真:0.1m.Equals((decimal)0.1)

最佳答案

我有一个 small source file它打印存储在 double 中的 exact 值。答案末尾的代码,以防链接消失。基本上,它获取 double 的确切位,然后从那里开始。它既不漂亮也不高效,但它确实有效:)

string s = DoubleConverter.ToExactString(0.1);
Console.WriteLine(s);

输出:

0.1000000000000000055511151231257827021181583404541015625

当您只使用 0.1.ToString() 时,BCL 会为您截断文本表示。

至于哪些 double 值可以精确表示 - 基本上,您需要计算出最接近的二进制表示是什么,并查看它是否是 的确切值.基本上它需要在正确的范围和精度内由2的幂(包括2的负幂)组成。

例如,4.75 可以精确表示为 22 + 2-1 + 2-2

源代码:

using System;
using System.Globalization;

/// <summary>
/// A class to allow the conversion of doubles to string representations of
/// their exact decimal values. The implementation aims for readability over
/// efficiency.
/// </summary>
public class DoubleConverter
{    
    /// <summary>
    /// Converts the given double to a string representation of its
    /// exact decimal value.
    /// </summary>
    /// <param name="d">The double to convert.</param>
    /// <returns>A string representation of the double's exact decimal value.</returns>
    public static string ToExactString (double d)
    {
        if (double.IsPositiveInfinity(d))
            return "+Infinity";
        if (double.IsNegativeInfinity(d))
            return "-Infinity";
        if (double.IsNaN(d))
            return "NaN";

        // Translate the double into sign, exponent and mantissa.
        long bits = BitConverter.DoubleToInt64Bits(d);
        // Note that the shift is sign-extended, hence the test against -1 not 1
        bool negative = (bits < 0);
        int exponent = (int) ((bits >> 52) & 0x7ffL);
        long mantissa = bits & 0xfffffffffffffL;

        // Subnormal numbers; exponent is effectively one higher,
        // but there's no extra normalisation bit in the mantissa
        if (exponent==0)
        {
            exponent++;
        }
        // Normal numbers; leave exponent as it is but add extra
        // bit to the front of the mantissa
        else
        {
            mantissa = mantissa | (1L<<52);
        }
        
        // Bias the exponent. It's actually biased by 1023, but we're
        // treating the mantissa as m.0 rather than 0.m, so we need
        // to subtract another 52 from it.
        exponent -= 1075;
        
        if (mantissa == 0) 
        {
            return "0";
        }
        
        /* Normalize */
        while((mantissa & 1) == 0) 
        {    /*  i.e., Mantissa is even */
            mantissa >>= 1;
            exponent++;
        }
        
        /// Construct a new decimal expansion with the mantissa
        ArbitraryDecimal ad = new ArbitraryDecimal (mantissa);
        
        // If the exponent is less than 0, we need to repeatedly
        // divide by 2 - which is the equivalent of multiplying
        // by 5 and dividing by 10.
        if (exponent < 0) 
        {
            for (int i=0; i < -exponent; i++)
                ad.MultiplyBy(5);
            ad.Shift(-exponent);
        } 
        // Otherwise, we need to repeatedly multiply by 2
        else
        {
            for (int i=0; i < exponent; i++)
                ad.MultiplyBy(2);
        }
        
        // Finally, return the string with an appropriate sign
        if (negative)
            return "-"+ad.ToString();
        else
            return ad.ToString();
    }
    
    /// <summary>Private class used for manipulating</summary>
    class ArbitraryDecimal
    {
        /// <summary>Digits in the decimal expansion, one byte per digit</summary>
        byte[] digits;
        /// <summary> 
        /// How many digits are *after* the decimal point
        /// </summary>
        int decimalPoint=0;

        /// <summary> 
        /// Constructs an arbitrary decimal expansion from the given long.
        /// The long must not be negative.
        /// </summary>
        internal ArbitraryDecimal (long x)
        {
            string tmp = x.ToString(CultureInfo.InvariantCulture);
            digits = new byte[tmp.Length];
            for (int i=0; i < tmp.Length; i++)
                digits[i] = (byte) (tmp[i]-'0');
            Normalize();
        }
        
        /// <summary>
        /// Multiplies the current expansion by the given amount, which should
        /// only be 2 or 5.
        /// </summary>
        internal void MultiplyBy(int amount)
        {
            byte[] result = new byte[digits.Length+1];
            for (int i=digits.Length-1; i >= 0; i--)
            {
                int resultDigit = digits[i]*amount+result[i+1];
                result[i]=(byte)(resultDigit/10);
                result[i+1]=(byte)(resultDigit%10);
            }
            if (result[0] != 0)
            {
                digits=result;
            }
            else
            {
                Array.Copy (result, 1, digits, 0, digits.Length);
            }
            Normalize();
        }
        
        /// <summary>
        /// Shifts the decimal point; a negative value makes
        /// the decimal expansion bigger (as fewer digits come after the
        /// decimal place) and a positive value makes the decimal
        /// expansion smaller.
        /// </summary>
        internal void Shift (int amount)
        {
            decimalPoint += amount;
        }

        /// <summary>
        /// Removes leading/trailing zeroes from the expansion.
        /// </summary>
        internal void Normalize()
        {
            int first;
            for (first=0; first < digits.Length; first++)
                if (digits[first]!=0)
                    break;
            int last;
            for (last=digits.Length-1; last >= 0; last--)
                if (digits[last]!=0)
                    break;
            
            if (first==0 && last==digits.Length-1)
                return;
            
            byte[] tmp = new byte[last-first+1];
            for (int i=0; i < tmp.Length; i++)
                tmp[i]=digits[i+first];
            
            decimalPoint -= digits.Length-(last+1);
            digits=tmp;
        }

        /// <summary>
        /// Converts the value to a proper decimal string representation.
        /// </summary>
        public override String ToString()
        {
            char[] digitString = new char[digits.Length];            
            for (int i=0; i < digits.Length; i++)
                digitString[i] = (char)(digits[i]+'0');
            
            // Simplest case - nothing after the decimal point,
            // and last real digit is non-zero, eg value=35
            if (decimalPoint==0)
            {
                return new string (digitString);
            }
            
            // Fairly simple case - nothing after the decimal
            // point, but some 0s to add, eg value=350
            if (decimalPoint < 0)
            {
                return new string (digitString)+
                       new string ('0', -decimalPoint);
            }
            
            // Nothing before the decimal point, eg 0.035
            if (decimalPoint >= digitString.Length)
            {
                return "0."+
                    new string ('0',(decimalPoint-digitString.Length))+
                    new string (digitString);
            }

            // Most complicated case - part of the string comes
            // before the decimal point, part comes after it,
            // eg 3.5
            return new string (digitString, 0, 
                               digitString.Length-decimalPoint)+
                "."+
                new string (digitString,
                            digitString.Length-decimalPoint, 
                            decimalPoint);
        }
    }
}

关于c# - 哪个 C# double 文字不能完全表示为 double?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24955647/

相关文章:

c# - 如何在 C# 中将 IEnumerable<T> 转换为 ConcurrentBag<T>?

c# - WPF 从 ResourceDictionary 读取样式到 C# 代码中的控件

C从函数传递双指针

java - 在 Java 中向 MatOfPoint 变量添加值

c++ - (-2147483648> 0) 在 C++ 中返回 true?

c++ - 为什么允许使用空的 wchar_t 文字?

types - 与文字的可区分联合

c# - 为什么.net framework 4.5安装后没有像之前版本一样更新注册表?

java - 如何在不转换为 Java 中的字符串的情况下将 double 字移动到末尾

c# - Microsoft Dynamics CRM SDK CRMServiceClient 连接字符串缓存错误