在数组或者集合中对自定义类型进行排序分为两种方法。

1.如果这个自定义类型是自己定义编写的,那么我可以使它继承ICompareable<T>接口,实现其中的CompareTo(Object)方法。然后直接Array.Sort(排序对象数组)对其进行排序。

     class Book: IComparable<Book>
{
//defined name and number for book
public string BookName { get; set; }
public string BookNo { get; set; }
// implement the CompareTo method
public int CompareTo(Book other)
{
if (other == null) throw new ArgumentNullException("other");
// compare to BookNo
return this.BookNo.CompareTo(other.BookNo);
}
}

我自定义了一个Book类型。有BookName和BookNo属性。我使它继承了ICompareable<Book>接口。并且实现了CompareTo(Book)方法。这样,我就可以直接用Array.Sort()对

这个数组按BookNo进行排序。

        static void test2()
{
Book[] bookArray = {
new Book{BookName = "AAA",BookNo = ""},
new Book{BookName = "DDD",BookNo = ""},
new Book{BookName = "CCC",BookNo = ""},
new Book{BookName = "BBB",BookNo = ""},
};
Array.Sort(bookArray);
foreach (Book item in bookArray)
{
Console.WriteLine("BookName = \"{0}\"; BookNo = \"{1}\".",item.BookName,item.BookNo);
} }

输出结果:

BookName = "AAA"; BookNo = "".
BookName = "BBB"; BookNo = "".
BookName = "CCC"; BookNo = "".
BookName = "DDD"; BookNo = "".

2.如果这个自定义类型不是自己编写的,是别人已经编写好的的一个类型,我不能修改这个类型。或者我想按照BookName排序,但是还不能修改现有的Book类该怎么办?

我们可以对这个类型进行包装。

     class Person
{
public string PersonName { get; set; }
public string PersonAge {get;set;}
}

Person这个类型没有继承ICompare接口。这个类不能修改,但是我还要对PersonAge进行排序。

我自己创建一个PersonCompare类,它实现了ICompare<T>接口,我把排序规则写在这个类中。

     class PersonCompare : IComparer<Person>
{
public int Compare(Person x, Person y)
{
if (x == null || y == null) throw new ArgumentNullException("argument error.");
return x.PersonAge.CompareTo(y.PersonAge); //sort rule
}
}

测试:

        static void test3()
{
Person[] personArray = {
new Person{PersonName = "AAA",PersonAge = ""},
new Person{PersonName = "EEE",PersonAge = ""},
new Person{PersonName = "CCC",PersonAge = ""},
new Person{PersonName = "FFF",PersonAge = ""},
};
Array.Sort(personArray,new PersonCompare());//second parameter is sort rule
foreach(Person item in personArray)
{
Console.WriteLine("PersonName = \"{0}\"; PersonAge = \"{1}\".", item.PersonName, item.PersonAge);
}
}

输出结果:

PersonName = "AAA"; PersonAge = "".
PersonName = "CCC"; PersonAge = "".
PersonName = "EEE"; PersonAge = "".
PersonName = "FFF"; PersonAge = "".

扩展:

如果我想对指定的属性进行排序怎么办?比如有的同事需要用PersonAge进行排序,有的需要使用PersonName进行排序。这种需求很常见。我们修改下PersonCompare方法。

为了使代码更加的规范。我建议以Person的属性为基础创建一个枚举。这个enum控制着我要按照那个属性进行排序。

    enum PersonType
{
PersonName,
PersonAge
}

我们需要PersonType作为参数传递给PersonCompare。以实现根据需求来定制排序规则。

   class PersonCompare : IComparer<Person>
{
private PersonType useType;
public PersonCompare(PersonType pt)
{
this.useType = pt;
}
public int Compare(Person x, Person y)
{
if (x == null || y == null) throw new ArgumentNullException("argument error.");
//return x.PersonAge.CompareTo(y.PersonAge);
switch (useType){
case PersonType.PersonAge:
return x.PersonAge.CompareTo(y.PersonAge);
case PersonType.PersonName:
return x.PersonName.CompareTo(y.PersonName);
default :
throw new ArgumentNullException("Doesn't contain this type.");
}
}
}

测试:

        static void test3()
{
Person[] personArray = {
new Person{PersonName = "AAA",PersonAge = ""},
new Person{PersonName = "EEE",PersonAge = ""},
new Person{PersonName = "CCC",PersonAge = ""},
new Person{PersonName = "FFF",PersonAge = ""},
};
Array.Sort(personArray,new PersonCompare(PersonType.PersonAge));// sort by age
foreach(Person item in personArray)
{
Console.WriteLine("PersonName = \"{0}\"; PersonAge = \"{1}\".", item.PersonName, item.PersonAge);
}
Console.WriteLine("---------------------------------------------------");
Array.Sort(personArray, new PersonCompare(PersonType.PersonName)); // sort by name
foreach (Person item in personArray)
{
Console.WriteLine("PersonName = \"{0}\"; PersonAge = \"{1}\".", item.PersonName, item.PersonAge);
}
}

输出结果:

PersonName = "CCC"; PersonAge = "".
PersonName = "AAA"; PersonAge = "".
PersonName = "EEE"; PersonAge = "".
PersonName = "FFF"; PersonAge = "".
---------------------------------------------------
PersonName = "AAA"; PersonAge = "".
PersonName = "CCC"; PersonAge = "".
PersonName = "EEE"; PersonAge = "".
PersonName = "FFF"; PersonAge = "".

总结:

其实数组和集合的排序一样。如果对自己定义的类型数组或者集合排序就用IComareable<T>。如果要对已有的类型数组或者集合排序就用IComare<T>.

C#自定义类型数组排序的更多相关文章

  1. 《精通C#》自定义类型转化-扩展方法-匿名类型-指针类型(11.3-11.6)

    1.类型转化在C#中有很多,常用的是int类型转string等,这些都有微软给我们定义好的,我们需要的时候直接调用就是了,这是值类型中的转化,有时候我们还会需要类类型(包括结构struct)的转化,还 ...

  2. C#简单问题,不简单的原理:不能局部定义自定义类型(不含匿名类型)

    今天在进行代码测试时发现,尝试在一个方法中定义一个委托,注意是定义一个委托,而不是声明一个委托变量,在编写的时候没有报错,VS也能智能提示,但在编译时却报语法不完整,缺少方括号,但实际查询并没有缺少, ...

  3. Struts2框架的自定义类型转换器

    前言:对于java的基本数据类型及一些系统类(如Date类.集合类),Struts2提供了内置类型转换功能,但是也有一定的限制.所以就演示出自定义类型转换器 一.应用于局部类型转换器 eg.用户登录出 ...

  4. sruts2 自定义类型转换器

    1.1.1    Struts2中自定义类型转换器:(了解) 类型转换的过程是双向的过程: JSP---->Action参数提交:String---Date. Action---->JSP ...

  5. 一个关于自定义类型作为HashMap的key的问题

    在之前的项目需要用到以自定义类型作为HashMap的key,遇到一个问题:如果修改了已经存储在HashMap中的实例,会发生什么情况呢?用一段代码来试验: import java.util.HashM ...

  6. Struts2之自定义类型转换器

    Struts2自定义类型转换器分为局部类型转换器和全局类型转换器 (1)局部类型转换器 如果页面传来一个参数reg.action?birthday=2010-11-12到后台action,然后属性用d ...

  7. [原创]java WEB学习笔记67:Struts2 学习之路-- 类型转换概述, 类型转换错误修改,如何自定义类型转换器

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  8. 自定义类型转换器converter

    作用:目前将日期转换成string,将string转换成我想要的类型   0509课件里讲 一.数据类型转换在web应用程序中,数据存在两个方向上的转换:1.当提交表单时  表单数据以字符串的形式提交 ...

  9. 用SQLData读写数据库自定义类型

    如何读写自定义类型?SQLData是个很直观的解决办法 在oracle使用手册上找到了很好的资料 点击打开链接 http://docs.oracle.com/cd/B10501_01/java.920 ...

随机推荐

  1. Dropbox面向第三方开发者推出全新的Datastore API

    Dropbox今天推出了全新的高级的同步API,开发者可以使用Dropbox的技术同步跨设备app的数据. Datastore API在现有的Dropbox Sync API基础上进行了扩展,允许开发 ...

  2. django 报错 no such table: auth_user

    需要执行 python3 manage.py makemigrations python3 manage.py migrate 参考:http://arrayoverflow.com/question ...

  3. Linux内存管理【转】

    转自:http://www.cnblogs.com/wuchanming/p/4360264.html 转载:http://www.kerneltravel.net/journal/v/mem.htm ...

  4. inline关键词的使用(转载)

    (一)inline函数(摘自C++ Primer的第三版) 在函数声明或定义中函数返回类型前加上关键字inline即把min()指定为内联. inline int min(int first, int ...

  5. 浅谈HookSSDT和和Resume(恢复)SSDT

     最近在学HookSSDT和针对Hook的ResumeSSDT,避免自己理解有所偏差,把它们写出来,希望大家不吝赐教.(虽然已经是过时了的技术,但是最起码了解其中的原理,嘿嘿嘿.) 转载注明出处:ht ...

  6. 微信小程序踩坑之一[thist]使用技巧

    刚上手小程序 时,习惯把this当成jquery中的$(this)来用,实际上这两个还是有差别的 在页面方法中调用其他方法,一般是用this.function(),直接调用小程序 的方法或函数则是用w ...

  7. BZOJ 4540 [Hnoi2016]序列 (单调栈 + ST表 + 莫队算法)

    题目链接  BZOJ4540 考虑莫队算法. 这题难在$[l, r]$到$[l, r+1]$的转移. 根据莫队算法的原理,这个时候答案应该加上 $cal(l, r+1) + cal(l+1, r+1) ...

  8. 松鼠的新家(lca)

    洛谷—— P3258 [JLOI2014]松鼠的新家 题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的 ...

  9. FMDB支持的事务类型

    FMDB支持的事务类型   在数据库中,事务可以保证数据操作的完整性.当存在大量并发操作,容易出现死锁问题.在SQLite中,为了解决该问题,提供三种事务模式,分别为DEFFERED.IMMEDIAT ...

  10. Office 各版本下载链接

    Office 2007 链接: https://pan.baidu.com/s/1pNJDlafw6KQSlljRUAQtWw 提取码: xoml 密钥:DBXYD-TF477-46YM4-W74MH ...