[英]How to use a custom comparer for EqualityComparer<T>.Default?
[英]How does EqualityComparer<T>.Default works internally?
考慮 T = 字符串。
我很好奇它是否使用類似的東西: typeof(EqualityComparer<T>).GetInterface("IEqualityComparer<T>");
有什么建議..
反射器提供:
public static EqualityComparer<T> Default
{
get
{
EqualityComparer<T> defaultComparer = EqualityComparer<T>.defaultComparer;
if (defaultComparer == null)
{
defaultComparer = EqualityComparer<T>.CreateComparer();
EqualityComparer<T>.defaultComparer = defaultComparer;
}
return defaultComparer;
}
}
private static EqualityComparer<T> CreateComparer()
{
RuntimeType c = (RuntimeType) typeof(T);
if (c == typeof(byte))
{
return (EqualityComparer<T>) new ByteEqualityComparer();
}
if (typeof(IEquatable<T>).IsAssignableFrom(c))
{
return (EqualityComparer<T>) RuntimeTypeHandle.CreateInstanceForAnotherGenericParameter((RuntimeType) typeof(GenericEqualityComparer<int>), c);
}
if (c.IsGenericType && (c.GetGenericTypeDefinition() == typeof(Nullable<>)))
{
RuntimeType type2 = (RuntimeType) c.GetGenericArguments()[0];
if (typeof(IEquatable<>).MakeGenericType(new Type[] { type2 }).IsAssignableFrom(type2))
{
return (EqualityComparer<T>) RuntimeTypeHandle.CreateInstanceForAnotherGenericParameter((RuntimeType) typeof(NullableEqualityComparer<int>), type2);
}
}
if (c.IsEnum && (Enum.GetUnderlyingType(c) == typeof(int)))
{
return (EqualityComparer<T>) RuntimeTypeHandle.CreateInstanceForAnotherGenericParameter((RuntimeType) typeof(EnumEqualityComparer<int>), c);
}
return new ObjectEqualityComparer<T>();
}
所以你可以看到如果 T = string 它會返回GenericEqualityComparer<string>
。
EqualityComparer<T>.Default
通過調用由System.Object
定義但可能會或可能不會被T
覆蓋的virtual
方法Equals(object)
和GetHashCode()
。
請注意,由於方法是virtual
,因此可以使用比T
更派生的類的實現。 例如:
EqualityComparer<object>.Default
.Equals(new Uri("http://example.com/"), new Uri("http://example.com/"))
將返回true
,即使
Object.ReferenceEquals(new Uri("http://example.com/"), new Uri("http://example.com/"))
和
(object)new Uri("http://example.com/") == (object)new Uri("http://example.com/")
都返回false
。
在T
是string
的情況下,類System.String
重載了有問題的兩個方法並使用序數比較。 因此在這種情況下它應該等效於System.StringComparer.Ordinal
。 當然, string
是一個sealed
類,所以沒有其他類可以從string
派生並以某種奇怪的方式覆蓋Equals
和GetHashCode
。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.