C# IComparable接口、IComparer接口和CompareTo(Object x)方法、Compare()方法

在项目中经常会用到字符串比较,但是有时候对字符串的操作比较多,规则各异。比如有的地方我们需要用排序规则,有的地方需要忽略大小写,我们该如何写一个比较容易操作的比较方法呢?重新实现IComparer接口不失为一个好办法。

 

 

 

IComparable.CompareTo 方法

在MSDN上是这么解释(机器翻译过来)的:

IComparable接口:定义一种特定于类型的通用比较方法,值类型或类通过实现此方法对其实例进行排序。

IComparer接口:公开一种比较两个对象的方法。

 

详细理解就是:

在默认情况下,对象的Equals(object o)方法(基类Object提供),是比较两个对象变量是否引用同一对象。

我们要必须我们自己的对象,必须自己定义对象比较方式。

IComparable和ICompare 接口是.net framework 中比较对象的标准方式,这两个接口之间的区别如下:

IComparable是标识这个类型具有比较的能力,可以比较该对象和另一个对象。一般情况下,我们使用 IComparable 给出类的默认比较代码,使用其他类给出非默认的比较代码。中文应该叫“可比较对象”。

IComparer是在一个单独的类中实现,是指当前对象可以用来比较任意两个对象,而当前对象本身不参与计算,中文应该叫“比较器”。

CompareTo:IComparable提供了一个方法int CompareTo(object obj)。这个方法接受一个对象,将当前实例的对象与同一类型的另一个对象进行比较,并返回一个整数,该整数指示当前实例在排序顺序中的位置是位于另一个对象之前、之后还是与其位置相同。

Compare:IComparer 也提供了一个方法 Compare().这个方法接受两个对象,返回一个整型结果,这与 CompareTo()相同。

 

IComparable接口

比如:有一个默认实例对象person1,需要和person2判断,判断person1是否比person2更年老或年轻,实际上,这个方法返回一个int,所以可和下面的代码说明person1更年老还是更年轻。

复制代码
 1  if(person1.CompareTo(person2) == 0)
 2  {
 3    Console.WriteLine("Same age");//年龄一样,person1和person2比较后返回0
 4  }
 5  else if(person1.CompareTo(person2) > 0 )
 6  {
 7    Console.WriteLine("person1 is older");//如果person1>person2,person1和person2比较后返回正整数,person1比person2年龄大
 8  }
 9  else
10 {
11    Console.WriteLine("person1 is younger");//如上述if不成立,person1<person2,person1和person2比较后返回正整数,person1比person2年龄小
12  }
复制代码

 

IComparer接口 

同样适用person来做例子

比如:有一个对象person1,需要和person2判断,判断person1是否比person2更年老或年轻,实际上,这个方法同样也是返回一个int,但是我们使用的是另外的接口(IComparer)的Compare来写,所以可和下面的代码说明person1更年老还是更年轻。

复制代码
 1 if(personComparer.Compare(person1,person2) == 0)
 2 {
 3   Console.WriteLine("same age");
 4 }
 5 else if(personComparer.Compare(person1,person2) > 0 )
 6 {
 7   Console.WriteLine("person 1 is older");
 8 }
 9 else
10 {
11   Console.WriteLine("person1 is younger");
12 }
复制代码

在这两种情况下,提供给方法的参数(person1、person2)是system.object类型。也就是说,可以比较任意类型的两个对象。所以,在返回结果之前,通常需要进行某种类型比较,如果使用了错误的类型,还会抛出异常。实际上,我们是使用泛型接口IComparable<T>,可以省略对象转换。

对象之间的大小比较,在开发过程中用的还是很多的,比如:string是一个可比较对象,它具有和另外一个String比较的能力,它会使用字母序进行比较。但是,有时候我们的一些逻辑需要一些特殊的比较方法。例如:WindowsExplorer 中的文件排序会认为”File10.txt" > "File2.txt"。这时候,我们就需要写一个特殊的IComparer对象实现这些特定的逻辑。

 

-----------------------------------------------------------------------------分割线-----------------------------------------------------------------------------

 

IComparable.CompareTo(Object x)的比较方式:

compareTo()的返回值是整型类型,它是先比较对应字符的大小(ASCII码顺序),如果第一个字符和另一个参数的第一个字符不等,结束比较,返回他们ASCII码之间的差值,如果第一个字符和另一个参数的第一个字符相等,则以第二个

字符和另一个参数的第二个字符做比较,以此类推,直至比较的字符或被比较的字符有一方全比较完,这时就比较字符的长度. 

举个Java的例子说明:

String s1 = "abc"; 
String s2 = "abcd"; 
String s3 = "abcdefg"; 
String s4 = "1bcdfg"; 
String s5 = "cdfg"; 
System.out.println( s1.compareTo(s2) ); // -1 (前面相等,s1长度小1) 
System.out.println( s1.compareTo(s3) ); // -4 (前面相等,s1长度小4) 
System.out.println( s1.compareTo(s4) ); // 48 ("a"的ASCII码是97,"1"的的ASCII码是49,所以返回48) 
System.out.println( s1.compareTo(s5) ); // -2 ("a"的ASCII码是97,"c"的ASCII码是99,所以返回-2)

 -----------------------------------------------------------------------------分割线-----------------------------------------------------------------------------

C# 如何实现compareTo()方法

那么在什么时候使用Icomparable.CompareTo(object obj)方法呢?

如果一个类实现了Icomparable接口,就可以重写Icomparable接口中的CompareTo(object obj)方法,假设集合中有几个学员对象,要按学员的姓名进行排序,那么就可以重写CompareTo方法按姓名排序,但如果又想按年龄排序了怎么办?CompareTo方法只能重写一次啊,这时就可以用到Icompare了。Icompare就叫做比较器,集合默认的Sort()排序方法有种重载的参数就是Icompare比较器对象。

像所有集合类一样,它允许您对所有实现了IComparable接口的对象进行排序。在下一个例子中,您将修改NameCompare类以实现IComparable的泛型接口:

public class NameCompare: IComparable<Student>

实现IComparable<Student>接口,NameCompare对象必须提供CompareTo()方法:

复制代码
 1 class NameCompare:IComparer<Student>     //姓名升序
 2 {
 3    public int Compare(Student x, Student y) 
 4    {
 5         return x.Name.CompareTo(y.Name);
 6    }
 7 }
 8 
 9 class NameCompare1 : IComparer<Student>     //姓名降序
10 {
11     public int Compare(Student x, Student y)
12     {
13         return y.Name.CompareTo(x.Name);
14     }
15 }
16 
17 class AgeCompare : IComparer<Student> 
18 {
19     public int Compare(Student x, Student y) 
20     {
21         return x.Age.CompareTo(y.Age);
22     }
23 }
复制代码

要排序的时候:

复制代码
 1 private void button3_Click(object sender, EventArgs e)
 2 {
 3     init();
 4     students.Sort(new NameCompare());               //姓名升序,Sort方法的参数就是实现了Icompare接口的类的对象
 5     show(students);
 6 }
 7 
 8 private void button4_Click(object sender, EventArgs e)
 9 {
10     init();
11     students.Sort(new NameCompare1());          //姓名降序
12     show(students);
13 }
14 
15 private void button5_Click(object sender, EventArgs e)
16 {
17     init();
18     students.Sort(new AgeCompare());
19     show(students);
20 }
posted @ 2021-09-11 23:39  小林野夫  阅读(990)  评论(0编辑  收藏  举报
原文链接:https://www.cnblogs.com/cdaniu/