我有这个代码我在linqpad中运行:
long x = long.MaxValue;
decimal y = x;
x.Dump();
y.Dump();
(x == y).Dump();
(y == x).Dump();
Object.Equals(x, y).Dump();
Object.Equals(y, x).Dump();
x.Equals(y).Dump();
y.Equals(x).Dump();
它产生这个输出:
9223372036854775807
9223372036854775807
True
True
False
False
False
True
注意最后两行:x.Equals(y)为false但y.Equals(x)为true。因此,十进制认为自身等于具有相同值的long,但long不认为自己等于具有相同值的十进制。
这种行为的解释是什么?
更新
我接受了Lee的回答。
我对此非常好奇并写了这个小程序:
using System;
namespace TestConversion
{
class Program
{
static void Main(string[] args)
{
long x = long.MaxValue;
decimal y = x;
Console.WriteLine(x);
Console.WriteLine(y);
Console.WriteLine(x == y);
Console.WriteLine(y == x);
Console.WriteLine(Object.Equals(x, y));
Console.WriteLine(Object.Equals(y, x));
Console.WriteLine(x.Equals(y));
Console.WriteLine(y.Equals(x));
Console.ReadKey();
}
}
}
然后我在IL中反汇编:
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
.maxstack 2
.locals init (
[0] int64 x,
[1] valuetype [mscorlib]System.Decimal y)
L_0000: nop
L_0001: ldc.i8 9223372036854775807
L_000a: stloc.0
L_000b: ldloc.0
L_000c: call valuetype [mscorlib]System.Decimal [mscorlib]System.Decimal::op_Implicit(int64)
L_0011: stloc.1
L_0012: ldloc.0
L_0013: call void [mscorlib]System.Console::WriteLine(int64)
L_0018: nop
L_0019: ldloc.1
L_001a: call void [mscorlib]System.Console::WriteLine(valuetype [mscorlib]System.Decimal)
L_001f: nop
L_0020: ldloc.0
L_0021: call valuetype [mscorlib]System.Decimal [mscorlib]System.Decimal::op_Implicit(int64)
L_0026: ldloc.1
L_0027: call bool [mscorlib]System.Decimal::op_Equality(valuetype [mscorlib]System.Decimal, valuetype [mscorlib]System.Decimal)
L_002c: call void [mscorlib]System.Console::WriteLine(bool)
L_0031: nop
L_0032: ldloc.1
L_0033: ldloc.0
L_0034: call valuetype [mscorlib]System.Decimal [mscorlib]System.Decimal::op_Implicit(int64)
L_0039: call bool [mscorlib]System.Decimal::op_Equality(valuetype [mscorlib]System.Decimal, valuetype [mscorlib]System.Decimal)
L_003e: call void [mscorlib]System.Console::WriteLine(bool)
L_0043: nop
L_0044: ldloc.0
L_0045: box int64
L_004a: ldloc.1
L_004b: box [mscorlib]System.Decimal
L_0050: call bool [mscorlib]System.Object::Equals(object, object)
L_0055: call void [mscorlib]System.Console::WriteLine(bool)
L_005a: nop
L_005b: ldloc.1
L_005c: box [mscorlib]System.Decimal
L_0061: ldloc.0
L_0062: box int64
L_0067: call bool [mscorlib]System.Object::Equals(object, object)
L_006c: call void [mscorlib]System.Console::WriteLine(bool)
L_0071: nop
L_0072: ldloca.s x
L_0074: ldloc.1
L_0075: box [mscorlib]System.Decimal
L_007a: call instance bool [mscorlib]System.Int64::Equals(object)
L_007f: call void [mscorlib]System.Console::WriteLine(bool)
L_0084: nop
L_0085: ldloca.s y
L_0087: ldloc.0
L_0088: call valuetype [mscorlib]System.Decimal [mscorlib]System.Decimal::op_Implicit(int64)
L_008d: call instance bool [mscorlib]System.Decimal::Equals(valuetype [mscorlib]System.Decimal)
L_0092: call void [mscorlib]System.Console::WriteLine(bool)
L_0097: nop
L_0098: call valuetype [mscorlib]System.ConsoleKeyInfo [mscorlib]System.Console::ReadKey()
L_009d: pop
L_009e: ret
}
您确实可以看到长值转换为十进制。
谢谢你们!
答案 0 :(得分:18)
这是因为
y.Equals(x);
正在调用decimal.Equals(decimal)
重载,因为long
和decimal
之间存在implicit conversion。结果,比较返回true。
但是,由于没有从decimal
到long
x.Equals(y)
调用long.Equals(object)
会导致y
被装箱,并且比较会返回false
,因为无法将其取消装箱。
答案 1 :(得分:6)
隐式转换和显式转换。
来自MSDN:
隐式转换:不需要特殊语法,因为 转换是类型安全的,不会丢失任何数据。例子包括 转换从较小到较大的整数类型和转换 从派生类到基类。
显式转换(强制转换):显式转换需要强制转换 运营商。当信息可能丢失时,需要进行强制转换 转换,或转换可能对其他转换不成功 原因。典型的例子包括数字转换为一种类型 具有较低的精度或较小的范围,以及基类的转换 实例到派生类。
long很容易转换为小数,但反之则不然,因此评估失败。
答案 2 :(得分:-2)
您正在比较对象引用和值。当然,参考文献并不相同 - 除了你在第2行明确设置的参考文献。但值是。
C#自动为您处理指针管理(例如“引用内存”)。您正在访问该层,该层最初从语法中看不出来。这是C#的本质。