我一直在寻找结构的最重要的指导方针,但我能找到的只是课程。
起初我以为我不必检查传递的对象是否为null,因为结构是值类型而不能为null。但现在我开始想到它,因为平等签名是
public bool Equals(object obj)
似乎没有什么能阻止我的struct的用户尝试将它与任意引用类型进行比较。
我的第二点涉及在我比较我的结构中的私有字段之前我想(我想)必须做的转换。我该如何将对象强制转换为struct的类型? C#的as
关键字似乎只适用于参考类型。
答案 0 :(得分:72)
struct MyStruct
{
public override bool Equals(object obj)
{
if (!(obj is MyStruct))
return false;
MyStruct mys = (MyStruct) obj;
// compare elements here
}
}
答案 1 :(得分:13)
我想,如果一个人使用.NET 4.5 ,可以使用documentation中所述的默认实现:
定义自己的类型时,该类型继承其基类型的Equals方法定义的功能。
ValueType.Equals :值相等;要么直接逐字节比较,要么使用反射进行逐场比较。
答案 2 :(得分:6)
如果有人想知道在Nullable对象中绑定结构的性能损失(为了避免来自is
和强制转换的双重类型检查),那么 是不可忽视的开销。
tl; dr :使用is
&在这种情况下施展。
struct Foo : IEquatable<Foo>
{
public int a, b;
public Foo(int a, int b)
{
this.a = a;
this.b = b;
}
public override bool Equals(object obj)
{
#if BOXING
var obj_ = obj as Foo?;
return obj_ != null && Equals(obj_.Value);
#elif DOUBLECHECK
return obj is Foo && Equals((Foo)obj);
#elif MAGIC
?
#endif
}
public bool Equals(Foo other)
{
return a == other.a && b == other.b;
}
}
class Program
{
static void Main(string[] args)
{
RunBenchmark(new Foo(42, 43), new Foo(42, 43));
RunBenchmark(new Foo(42, 43), new Foo(43, 44));
}
static void RunBenchmark(object x, object y)
{
var sw = Stopwatch.StartNew();
for (var i = 0; i < 100000000; i++) x.Equals(y);
sw.Stop();
Console.WriteLine(sw.ElapsedMilliseconds);
}
}
结果:
BOXING
EQ 8012 7973 7981 8000
NEQ 7929 7715 7906 7888
DOUBLECHECK
EQ 3654 3650 3638 3605
NEQ 3310 3301 3319 3297
警告:此测试可能在很多方面存在缺陷,但我确实验证了基准代码本身并未以奇怪的方式进行优化。
看看IL,双重检查方法编译得更清洁。
拳击IL:
.method public hidebysig virtual
instance bool Equals (
object obj
) cil managed
{
// Method begins at RVA 0x2060
// Code size 37 (0x25)
.maxstack 2
.locals init (
[0] valuetype [mscorlib]System.Nullable`1<valuetype StructIEqualsImpl.Foo> obj_
)
IL_0000: ldarg.1
IL_0001: isinst valuetype [mscorlib]System.Nullable`1<valuetype StructIEqualsImpl.Foo>
IL_0006: unbox.any valuetype [mscorlib]System.Nullable`1<valuetype StructIEqualsImpl.Foo>
IL_000b: stloc.0
IL_000c: ldloca.s obj_
IL_000e: call instance bool valuetype [mscorlib]System.Nullable`1<valuetype StructIEqualsImpl.Foo>::get_HasValue()
IL_0013: brfalse.s IL_0023
IL_0015: ldarg.0
IL_0016: ldloca.s obj_
IL_0018: call instance !0 valuetype [mscorlib]System.Nullable`1<valuetype StructIEqualsImpl.Foo>::get_Value()
IL_001d: call instance bool StructIEqualsImpl.Foo::Equals(valuetype StructIEqualsImpl.Foo)
IL_0022: ret
IL_0023: ldc.i4.0
IL_0024: ret
} // end of method Foo::Equals
仔细检查IL:
.method public hidebysig virtual
instance bool Equals (
object obj
) cil managed
{
// Method begins at RVA 0x2060
// Code size 23 (0x17)
.maxstack 8
IL_0000: ldarg.1
IL_0001: isinst StructIEqualsImpl.Foo
IL_0006: brfalse.s IL_0015
IL_0008: ldarg.0
IL_0009: ldarg.1
IL_000a: unbox.any StructIEqualsImpl.Foo
IL_000f: call instance bool StructIEqualsImpl.Foo::Equals(valuetype StructIEqualsImpl.Foo)
IL_0014: ret
IL_0015: ldc.i4.0
IL_0016: ret
} // end of method Foo::Equals
向罗曼·雷纳(Roman Reiner)道具,发现了一个真正让我看起来不太好的错误。
答案 3 :(得分:6)
感谢some news in C# 7.0,有一种更简单的方法来完成与接受的答案相同的方法:
n
或者我最喜欢的 - 与表达身体功能相同:
struct MyStruct
{
public override bool Equals(object obj)
{
if (!(obj is MyStruct mys)) // type pattern here
return false;
return this.field1 == mys.field1 && this.field2 == mys.field2 // mys is already known here without explicit casting
}
}
答案 4 :(得分:5)
使用is
运算符:
public bool Equals(object obj)
{
if (obj is MyStruct)
{
var o = (MyStruct)obj;
...
}
}
答案 5 :(得分:0)
添加现有答案。
如果附加一个,你仍然可以拥有可以为空的值吗?在struct name之后(这适用于每个值对象)
int?
通过调用(MyStructName)variableName