c# – 如何确定两个通用类型值是否相等?

前端之家收集整理的这篇文章主要介绍了c# – 如何确定两个通用类型值是否相等?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
*更新
我很抱歉…我的示例代码包含一个错误,导致了很多我不明白的答案.
代替
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.

所以这里有两个问题:

>为什么代码不返回true?
>如何实现Equals方法,以确保返回true?

解决方法

Marc Gravell’s answer所示,问题是与Equals(StringBuilder)不同的StringBuilder Equals(object)实现.

然后,您可以忽略该问题,因为您的代码将与任何其他一致执行的类一起工作,或者您可以使用动态来解决问题(再次按照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);
}

猜你在找的C#相关文章