Is .NET “decimal” arithmetic independent of platform/architecture?

后端 未结 4 925
栀梦
栀梦 2021-01-04 03:13

I asked about System.Double recently and was told that computations may differ depending on platform/architecture. Unfortunately, I cannot find any information

4条回答
  •  北荒
    北荒 (楼主)
    2021-01-04 03:53

    Am I guaranteed to get exactly the same result for any particular decimal computation independently of platform/architecture?

    The C# 4 spec is clear that the value you get will be computed the same on any platform.

    As LukeH's answer notes, the ECMA version of the C# 2 spec grants leeway to conforming implementations to provide more precision, so an implementation of C# 2.0 on another platform might provide a higher-precision answer.

    For the purposes of this answer I'll just discuss the C# 4.0 specified behaviour.

    The C# 4.0 spec says:


    The result of an operation on values of type decimal is that which would result from calculating an exact result (preserving scale, as defined for each operator) and then rounding to fit the representation. Results are rounded to the nearest representable value, and, when a result is equally close to two representable values, to the value that has an even number in the least significant digit position [...]. A zero result always has a sign of 0 and a scale of 0.


    Since the calculation of the exact value of an operation should be the same on any platform, and the rounding algorithm is well-defined, the resulting value should be the same regardless of platform.

    However, note the parenthetical and that last sentence about the zeroes. It might not be clear why that information is necessary.

    One of the oddities of the decimal system is that almost every quantity has more than one possible representation. Consider exact value 123.456. A decimal is the combination of a 96 bit integer, a 1 bit sign, and an eight-bit exponent that represents a number from -28 to 28. That means that exact value 123.456 could be represented by decimals 123456 x 10-3 or 1234560 x 10-4 or 12345600 x 10-5. Scale matters.

    The C# specification also mandates how information about scale is computed. The literal 123.456m would be encoded as 123456 x 10-3, and 123.4560m would be encoded as 1234560 x 10-4.

    Observe the effects of this feature in action:

    decimal d1 = 111.111000m;
    decimal d2 = 111.111m;
    decimal d3 = d1 + d1;
    decimal d4 = d2 + d2;
    decimal d5 = d1 + d2;
    Console.WriteLine(d1);
    Console.WriteLine(d2);
    Console.WriteLine(d3);
    Console.WriteLine(d4);
    Console.WriteLine(d5);
    Console.WriteLine(d3 == d4);
    Console.WriteLine(d4 == d5);
    Console.WriteLine(d5 == d3);
    

    This produces

    111.111000
    111.111
    222.222000
    222.222
    222.222000
    True
    True
    True
    

    Notice how information about significant zero figures is preserved across operations on decimals, and that decimal.ToString knows about that and displays the preserved zeroes if it can. Notice also how decimal equality knows to make comparisons based on exact values, even if those values have different binary and string representations.

    The spec I think does not actually say that decimal.ToString() needs to correctly print out values with trailing zeroes based on their scales, but it would be foolish of an implementation to not do so; I would consider that a bug.

    I also note that the internal memory format of a decimal in the CLR implementation is 128 bits, subdivided into: 16 unused bits, 8 scale bits, 7 more unused bits, 1 sign bit and 96 mantissa bits. The exact layout of those bits in memory is not defined by the specification, and if another implementation wants to stuff additional information into those 23 unused bits for its own purposes, it can do so. In the CLR implementation the unused bits are supposed to always be zero.

提交回复
热议问题