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

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. ACM程序设计选修课——Problem D: (ds:树)合并果子(最优二叉树赫夫曼算法)

    Problem D: (ds:树)合并果子 Time Limit: 5 Sec  Memory Limit: 64 MB Submit: 80  Solved: 4 [Submit][Status][ ...

  2. LibreOJ2241 - 「CQOI2014」排序机械臂

    Portal Description 给出一个\(n(n\leq10^5)\)个数的序列\(\{a_n\}\),对该序列进行\(n\)次操作.若在第\(i\)次操作前第\(i\)小的数在\(p_i\) ...

  3. 北京集训TEST13——PA(第k小数)

    题目: Description [问题描述] 从n个数中选若干(至少1)个数求和,求所有方案中第k小的和(和相同但取法不同的视为不同方案).[输入格式]    第一行输入2个正整数n,k.    第二 ...

  4. iOS-ARC机制

    内存管理是开发软件中重要的一个课题.如果内存管理不当,轻者内存泄露,重者程序崩溃. 下面重要讲述一下iOS的ARC(Automatic Reference Counting))机制. ARC的历史由来 ...

  5. Zabbix实现短信报警设置(实战)

    配置环境: zabbix 2.2.15 1.配置示警媒介类型 此文件所在位置:/usr/lib/zabbix/alertscripts/ 必须拥有执行权限,并且改变所属用户和组 要修改此脚本的路径,需 ...

  6. hdu 4301 dp

    Divide Chocolate Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  7. 58同城职位分类数据 json

    { "level0": {"0": "销售", "1": "客服", "2": ...

  8. JS函数调用原理($.ajax原理)

    参考四种函数调用:http://www.cnblogs.com/qlqwjy/p/7660013.html http://www.cnblogs.com/qlqwjy/p/7660375.html 1 ...

  9. 转 Python爬虫实战二之爬取百度贴吧帖子

    静觅 » Python爬虫实战二之爬取百度贴吧帖子 大家好,上次我们实验了爬取了糗事百科的段子,那么这次我们来尝试一下爬取百度贴吧的帖子.与上一篇不同的是,这次我们需要用到文件的相关操作. 本篇目标 ...

  10. C语言中的数组的访问方式

    闲下来,写的代码,很是简单,不解释,代码如下: #include <stdio.h> int main(int argc, char **argv) { char cArray[] = & ...