c#: List.Sort()实现稳固排序(stable sort)
1. 源起:
KV 7.0加入列表管理功能,处理排序问题时,对空列表执行按大小、日期、长度排序发现,其中次序会发生改变,令人纳闷。
没天理呀,不应该啊!List.Sort()方法,它为什么?
对此问题深入去了解,倒发现了有趣的问题:稳固排序与非稳固排序。
2、稳固排序与非稳固排序
在微软官方网站找到此段说明:
Remarks
If comparison is provided, the elements of the List<T> are sorted using the method represented by the delegate.
If comparison is null, an ArgumentNullException is thrown.
This method uses Array.Sort, which applies the introspective sort as follows:
If the partition size is fewer than 16 elements, it uses an insertion sort algorithm
If the number of partitions exceeds 2 * LogN, where N is the range of the input array, it uses a Heapsort algorithm.
Otherwise, it uses a Quicksort algorithm.
This implementation performs an unstable sort; that is, if two elements are equal, their order might not be preserved. In contrast, a stable sort preserves the order of elements that are equal.
On average, this method is an O(n log n) operation, where n is Count; in the worst case it is an O(n ^ 2) operation.
大意是此实现将执行不稳定排序。也就是说,如果两个元素相等,则可能不会保留其顺序。
我们建Demo做验证,用例如下:
var list = new List<string>();
list.AddRange(new string[] { "", "", "", "" }); list.Sort((x, y) =>
{
return ;
}); foreach (string s in list)
Console.WriteLine(s);
其输出为:

验证上述结果,推测它是使用了二分法快速反序,把后半部分置前了。
可八辈的我们宁愿它输入为原始顺序3214,这个合理,但是,可是,它无耻的变了……
3、实际场景
比如,我们需要对特定的对象进行排序,如下摘取网上例子(链接)稍加修正,以做示例:
static void Main(string[] args)
{
var p1 = new Person() { Name = "Abby", Age = };
var p4 = new Person() { Name = "Jason", Age = };
var p2 = new Person() { Name = "Bob", Age = };
var p3 = new Person() { Name = "Charlie", Age = };
var p5 = new Person() { Name = "Danielle", Age = }; var list = new List<Person>();
list.Add(p1);
list.Add(p2);
list.Add(p3);
list.Add(p4);
list.Add(p5);
list.Sort(); Console.WriteLine("Unstable List Sort:");
foreach (Person p in list)
Console.WriteLine(p); Console.ReadLine();
} class Person : IComparable
{
public string Name { get; set; }
public int Age { get; set; } public int CompareTo(object obj)
{
int result = ;
if (obj != null && obj is Person)
{
var person = (Person)obj;
result = this.Age.CompareTo(person.Age);;
}
return result;
} public override string ToString()
{
return String.Format("{0} - {1}", this.Name, this.Age);
}
}
其输出为:

其中间23岁之三项,没有按Jason、Bob、Charlie的次序来,甚至也不是倒序。
我就是想要它加入顺序不变,怎么办?
对此问题,搜出一些文章,有建议用LINQ排序因为它是稳固排序,可我们工程所用.net framework版本为2.0,不支持,只好做罢。
在stackoverflow上,碰到一群难兄难弟,诸般建议阅过,找到可用方法,加索引!
4、解决方法
且修改Person类,加入SortIndex属性如下,并修正其比较函数:
class Person : IComparable
{
public string Name { get; set; }
public int Age { get; set; }
public int SortIndex { get; set; }
public int CompareTo(object obj)
{
int result = ;
if (obj != null && obj is Person)
{
var person = (Person)obj;
result = this.Age.CompareTo(person.Age); if (result == )
result = this.SortIndex.CompareTo(person.SortIndex);
}
return result;
}
...
}
初始化时,加入其索引:
var p1 = new Person() { Name = "Abby", Age = , SortIndex = };
var p4 = new Person() { Name = "Jason", Age = , SortIndex = };
var p2 = new Person() { Name = "Bob", Age = , SortIndex = };
var p3 = new Person() { Name = "Charlie", Age = , SortIndex = };
var p5 = new Person() { Name = "Danielle", Age = , SortIndex = };
输出顺序:

保留初始顺序,解决问题。
后记:
之前未曾留意,此问题发现倒是很有意思,度娘不给力,解决问题还是要google,在stackoverflow上找到有同困扰之人,得其可参考方案。
此场景,通常用于自定义数据结构比较,如KV项目之空白列表,当其可比较项相等时,应该保留其原始顺序,可是被改变了,导致寻求解决问题的方法。
但在官方方案已固定不能改变之情况下,做个曲线救国,加另一索引以固定次序,虽然看来费些点工夫,但是能解决问题,不失为好的可用方法。
参考文档:
List(T).Sort Method (Comparison(T)) (System.Collections.Generic)
Why does List<T>.Sort method reorder equal IComparable<T> elements?
c#: List.Sort()实现稳固排序(stable sort)的更多相关文章
- C#中List调用库函数sort进行升序排序
private void button1_Click(object sender, EventArgs e) { List<int> demo2 = new List<int> ...
- python 排序之sort
#coding:utf-8 #求列表的第二大值 list_test =[6,2,4,6,1,2,3,4,5] list_test.sort() print list_test[-2] "&q ...
- 计数排序与桶排序(bucket sort)
Bucket Sort is a sorting method that subdivides the given data into various buckets depending on cer ...
- 数据结构杂谈(二)简单有趣的地精排序Gnome sort
很早之前便听说过地精排序的名字,今天自己看来一下,发现这是一种非常简单而且有趣的排序算法. 为什么叫地精排序? 地精排序在2000年由Dr. Hamid Sarbazi-Azad 提出的时候被称作 s ...
- sort 树 hash 排序
STL 中 sort 函数用法简介 做 ACM 题的时候,排序是一种经常要用到的操作.如果每次都自己写个冒泡之类的 O(n^2) 排序,不但程序容易超时,而且浪费宝贵的比赛时间,还很有可能写错. ST ...
- python 字典排序 关于sort()、reversed()、sorted()
一.Python的排序 1.reversed() 这个很好理解,reversed英文意思就是:adj. 颠倒的:相反的:(判决等)撤销的 print list(reversed(['dream','a ...
- (C++)STL排序函数sort和qsort的用法与区别
主要内容: 1.qsort的用法 2.sort的用法 3.qsort和sort的区别 qsort的用法: 原 型: void qsort(void *base, int nelem, int widt ...
- 排序命令sort
Unix和Linux自带的sort命令功能非常强大,其主要功能是对文本内容按不同的方法排序.它不仅可以按一个或多个字段排序,还可以合并文件.使用sort处理一些较大的文件时,可能处理速度会比较慢,但却 ...
- C++排序函数sort/qsort使用
问题描述: C++排序函数sort/qsort的使用 问题解决: (1)sort函数使用 注: sort函数,参数1为数组首地址,参数2是数组 ...
随机推荐
- eclipse工作区(workspace)常用设置(preferences)
切换工作区 新建一个作为工作区的文件夹 File -> Switch Workspace -> Other... -> browse,定位到新的指定工作区文件夹即可. 切换到新的工作 ...
- iOS响应链原理
ios找到被点击的view的过程是从根view开始递归地调用hitTest方法,直到有一个子view的hitTest方法返回自身:如果所有一级子view的hitTest方法都返回nil,那么根view ...
- 模糊查询内存查询java实现
下面说说看到的工作项目中的代码,是这个样子的,事先查询一次数据库,将查询到的整张表的数据存到内存,以后使用时不再查询数据库,而直接操作内存中的数据,这主要用于数据库中的数据比较稳定,不会轻易改变的情况 ...
- 在系统中使用Bean Validation验证参数
转自:http://www.importnew.com/18561.html 为什么要使用Bean Validation? 当我们实现某个接口时,都需要对入参数进行校验.例如下面的代码 1 2 3 ...
- LeetCode OJ 94. Binary Tree Inorder Traversal
Given a binary tree, return the inorder traversal of its nodes' values. For example:Given binary tre ...
- java中break和continue跳出指定循环(转载)
java中break和continue跳出指定循环 java中break和continue可以跳出指定循环,break和continue之后不加任何循环名则默认跳出其所在的循环,在其后加指定循环名,则 ...
- python 装饰器、递归原理、模块导入方式
1.装饰器原理 def f1(arg): print '验证' arg() def func(): print ' #.将被调用函数封装到另外一个函数 func = f1(func) #.对原函数重新 ...
- ArcGIS案例学习笔记4_2_城乡规划容积率计算和建筑景观三维动画
ArcGIS案例学习笔记4_2_城乡规划容积率计算和建筑景观三维动画 概述 计划时间:第4天下午 目的:城市规划容积率计算和建筑三维景观动画 教程: pdf page578 数据:实验数据\Chp13 ...
- Javascript中构造函数的返回值问题和new对象的过程
首先明确一点:javascript中构造函数是不需要有返回值的,这一点跟java很类似.可以认为构造函数和普通函数的最大差别就是:构造函数中没有return语句,普通函数可以有return语句:构造函 ...
- Android辅助开发工具说明
1.aapt(Android Asset Packaging Tool):用于建立zip包(zip.jar.apk),也可用于将资源编译到二进制的assets:2.adb(Android Debug ...