我很抱歉…我的示例代码包含一个错误,导致了很多我不明白的答案.
代替
Console.WriteLine("3. this.Equals " + (go1.Equals(go2)));
我打算写
Console.WriteLine("3. this.Equals " + (go1.Equals(sb2)));
我试图找出如何成功地确定两个通用类型值是否相等.基于Mark Byers在this question上的答案,我会认为我可以使用value.Equals(),其中value是泛型类型.
我的实际问题是在LinkedList实现中,但问题可以用这个更简单的例子来显示.
class GenericOjbect<T> { public T Value { get; private set; } public GenericOjbect(T value) { Value = value; } public bool Equals(T value) { return (Value.Equals(value)); } }
现在我定义一个GenericObject< StringBuilder>的实例包含新的StringBuilder(“StackOverflow”).如果我在这个GenericObject实例上调用Equals(新的StringBuilder(“StackOverflow”)),我会希望得到true,但是我得到false.
示例程序显示:
using System; using System.Text; class Program { static void Main() { var sb1 = new StringBuilder("StackOverflow"); var sb2 = new StringBuilder("StackOverflow"); Console.WriteLine("StringBuilder compare"); Console.WriteLine("1. == " + (sb1 == sb2)); Console.WriteLine("2. Object.Equals " + (Object.Equals(sb1,sb2))); Console.WriteLine("3. this.Equals " + (sb1.Equals(sb2))); var go1 = new GenericOjbect<StringBuilder>(sb1); var go2 = new GenericOjbect<StringBuilder>(sb2); Console.WriteLine("\nGenericObject compare"); Console.WriteLine("1. == " + (go1 == go2)); Console.WriteLine("2. Object.Equals " + (Object.Equals(go1,sb2))); Console.WriteLine("3. this.Equals " + (go1.Equals(sb2))); Console.WriteLine("4. Value.Equals " + (go1.Value.Equals(sb2.Value))); } }
对于比较两个StringBuilder对象的三种方法,只有StringBuilder.Equals实例方法(第3行)返回true.这就是我的预期.但是当比较GenericObject对象时,它的Equals()方法(第三行)返回false.有趣的是,第四个比较方法确实返回true.我认为第三次和第四次比较其实也是一样的.
我会预料到的.因为在GenericObject类的Equals()方法中,value和Value都是类型T,在这种情况下是StringBuilder.基于Mark Byers在this question中的答案,我期望Value.Equals()方法使用StringBuilder的Equals()方法.正如我所看到的,StringBuilder的Equal()方法确实返回true.
我甚至试过
public bool Equals(T value) { return EqualityComparer<T>.Default.Equals(Value,value); }
但也会返回false.
所以这里有两个问题:
解决方法
然后,您可以忽略该问题,因为您的代码将与任何其他一致执行的类一起工作,或者您可以使用动态来解决问题(再次按照Mark Gravell的建议).
但是,考虑到你没有使用C#4(所以没有动态),你可以这样尝试:
public bool Equals(T value) { // uses Reflection to check if a Type-specific `Equals` exists... var specificEquals = typeof(T).GetMethod("Equals",new Type[] { typeof(T) }); if (specificEquals != null && specificEquals.ReturnType == typeof(bool)) { return (bool)specificEquals.Invoke(this.Value,new object[]{value}); } return this.Value.Equals(value); }