在 C# 语言中提供了 IComparer 和 IComparable 接口比较集合中的对象值,主要用于对集合中的元素排序。
IComparer 接口用于在一个单独的类中实现,用于比较任意两个对象。
IComparable 接口用于在要比较的对象的类中实现,可以比较任意两个对象。
在比较器中还提供了泛型接口的表示形式,即 IComparer<T> 和 IComparable<T> 的形式。
对于 IComparer<T> 接口,方法如下表所示。
方法 | 作用 |
---|---|
CompareTo(T obj) | 比较两个对象值 |
如果需要对集合中的元素排序,通常使用 CompareTo 方法实现,下面通过实例来演示 CompareTo 方法的使用。
【实例 1】在上一节《C#泛型集合》中实例 1 的基础上将学生信息按照年龄从大到小输出。
根据题目要求,如果不使用比较器,由于集合中的元素是 Student 类型的,不能直接排序,需要按照 Student 学生信息类中的年龄属性排序,因此代码比较烦琐。
使用 CompareTo 方法实现比较简单。在 Student 类中添加 CompareTo 方法,代码如下。
- class Student:IComparable<Student>
- {
- //提供有参构造方法,为属性赋值
- public Student(int id,string name,int age)
- {
- this.id = id;
- this.name = name;
- this.age = age;
- }
- //学号
- public int id { get; set; }
- //姓名
- public string name { get; set; }
- //年龄
- public int age { get; set; }
- //重写ToString 方法
- public override string ToString()
- {
- return id + ":" + name + ":" + age;
- }
- //定义比较方法,按照学生的年龄比较
- public int CompareTo(Student other)
- {
- if (this.age > other.age)
- {
- return -1;
- }
- return 1;
- }
- }
在 Main 方法中创建泛型集合,并向集合中添加项以及进行排序的代码如下。
- class Program
- {
- static void Main(string[] args)
- {
- List<Student> list = new List<Student>();
- list.Add(new Student(1, "小明", 20));
- list.Add(new Student(2, "小李", 21));
- list.Add(new Student(3, "小赵", 22));
- list.Sort();
- foreach(Student stu in list)
- {
- Console.WriteLine(stu);
- }
- }
- }
执行上面的代码,效果如下图所示。
从上面的执行效果可以看出,在使用集合的 Sort 方法后,集合中的元素是按照学生年龄从大到小的顺序输出的。
在默认情况下,Sort 方法是将集合中的元素从小到大输出的, 由于在 Student 类中重写了 CompareTo 方法,因此会按照预先定义好的排序规则对学生信息排序。
需要说明的是,在 CompareTo 方法中返回值大于 0 则表示第一个对象的值大于第二个对象的值,返回值小于 0 则表示第一个对象的值小于第二个对象的值,返回值等于 0 则表示两个对象的值相等。
对于实例 1 中的操作也可以使用 IComparer<T> 接口来实现,IComparer<T> 接口中的方法如下表所示。
方法 | 作用 |
---|---|
Compare(T obj1,T obj2) | 比较两个对象值 |
在使用 IComparer<T> 接口中的 Compare 方法时,需要单独定义一个类来实现该比较方法。
下面通过实例演示 IComparer<T> 接口的使用。
【实例 2】将实例 1 用 IComparer<T> 接口实现。
根据题目要求,先定义一个比较器的类,再实现对集合中元素的排序,代码如下。
- class MyCompare : IComparer<Student>
- {
- //比较方法
- public int Compare(Student x,Student y)
- {
- if (x.age > y.age)
- {
- return -1;
- }
- return 1;
- }
- }
在 Main 方法中应用该比较器对集合中的元素排序,代码如下。
- class Program
- {
- static void Main(string[] args)
- {
- List<Student> list = new List<Student>();
- list.Add(new Student(1, "小明", 20));
- list.Add(new Student(2, "小李", 21));
- list.Add(new Student(3, "小赵", 22));
- //在Sort方法中传递自定义比较器作为参数
- list.Sort(new MyCompare);
- foreach(Student stu in list)
- {
- Console.WriteLine(stu);
- }
- }
- }
执行上面的代码,效果与实例 1 所示的一致。
从上面两个实例可以看出,不论使用 IComparer<T> 接口还是 IComparable<T> 接口都能自定义在集合中使用 Sort 方法时的排序。
提示:不仅在泛型集合中允许使用比较器,在非泛型集合中也允许使用比较器,并且可以使用非泛型接口的比较器。