C#自定义类型数组排序
在数组或者集合中对自定义类型进行排序分为两种方法。
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#自定义类型数组排序的更多相关文章
- 《精通C#》自定义类型转化-扩展方法-匿名类型-指针类型(11.3-11.6)
1.类型转化在C#中有很多,常用的是int类型转string等,这些都有微软给我们定义好的,我们需要的时候直接调用就是了,这是值类型中的转化,有时候我们还会需要类类型(包括结构struct)的转化,还 ...
- C#简单问题,不简单的原理:不能局部定义自定义类型(不含匿名类型)
今天在进行代码测试时发现,尝试在一个方法中定义一个委托,注意是定义一个委托,而不是声明一个委托变量,在编写的时候没有报错,VS也能智能提示,但在编译时却报语法不完整,缺少方括号,但实际查询并没有缺少, ...
- Struts2框架的自定义类型转换器
前言:对于java的基本数据类型及一些系统类(如Date类.集合类),Struts2提供了内置类型转换功能,但是也有一定的限制.所以就演示出自定义类型转换器 一.应用于局部类型转换器 eg.用户登录出 ...
- sruts2 自定义类型转换器
1.1.1 Struts2中自定义类型转换器:(了解) 类型转换的过程是双向的过程: JSP---->Action参数提交:String---Date. Action---->JSP ...
- 一个关于自定义类型作为HashMap的key的问题
在之前的项目需要用到以自定义类型作为HashMap的key,遇到一个问题:如果修改了已经存储在HashMap中的实例,会发生什么情况呢?用一段代码来试验: import java.util.HashM ...
- Struts2之自定义类型转换器
Struts2自定义类型转换器分为局部类型转换器和全局类型转换器 (1)局部类型转换器 如果页面传来一个参数reg.action?birthday=2010-11-12到后台action,然后属性用d ...
- [原创]java WEB学习笔记67:Struts2 学习之路-- 类型转换概述, 类型转换错误修改,如何自定义类型转换器
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- 自定义类型转换器converter
作用:目前将日期转换成string,将string转换成我想要的类型 0509课件里讲 一.数据类型转换在web应用程序中,数据存在两个方向上的转换:1.当提交表单时 表单数据以字符串的形式提交 ...
- 用SQLData读写数据库自定义类型
如何读写自定义类型?SQLData是个很直观的解决办法 在oracle使用手册上找到了很好的资料 点击打开链接 http://docs.oracle.com/cd/B10501_01/java.920 ...
随机推荐
- 学习C++的第四天
1.头文件中的 #ifndef/#define/#endif 防止该头文件被重复引用” //文件路径名:el_1\hello.h #ifndef _HELLO /////如果没有定义 _HELLO文件 ...
- 【bzoj1283】序列 线性规划与费用流
题目描述 给出一个长度为 的正整数序列Ci,求一个子序列,使得原序列中任意长度为 的子串中被选出的元素不超过K(K,M<=100) 个,并且选出的元素之和最大. 输入 第1行三个数N,m,k. ...
- HDU-3718 Similarity
题目只有26个字母,所以我们新建一个二分图,v[i][j]表示字母i对应字母j时能成功匹配的个数,给这个边矩阵v求个最大匹配就是答案. #include <cstdlib> #includ ...
- mybatis学习(六)——参数处理
先总结一下,后面再一个个解释: 单个参数:直接使用#{参数名}进行取值,mybatis没做特殊处理,参数名可以随便写. 多个参数:使用#{param1},#{param2}取值 命名参数:通过@par ...
- [NOIP2017] 时间复杂度 (模拟,栈)
题目链接 Solution 用栈进行模拟. 记录一个 \(map\) 来看循环变量有没有用过. 对于每一次入栈都加信息. 出栈直接将 \(top\) 减一下. 反正一堆乱七八糟的东西瞎搞... 注意条 ...
- bzoj1143/2718 祭祀river(最大独立集)
[CTSC2008]祭祀river Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2175 Solved: 1098[Submit][Status] ...
- 【前端学习笔记】2015-09-01 附 split()方法、readyState
1.split():作用对象是一个字符串或者字符串对象,会要求设置两个参数(分割点(separator),分割出来的数量(number)),ps:1."2:3:4:5".split ...
- ADO:用代码调用存储过程
原文发布时间为:2008-08-02 -- 来源于本人的百度文章 [由搬家工具导入] using System;using System.Data;using System.Configuration ...
- DBGridEH如何去表头?设置哪个属性
- Ui大屏
http://www.uimaker.com/plus/view.php?aid=128661&pageno=1