C#实现希尔排序
作者:Ruby_Lu 发布时间:2023-11-02 08:15:04
对于大规模乱序的数组,插入排序很慢,因为它只会交换相邻的元素,因此元素只能一点一点地从数组地一段移动到另一端。希尔排序改进了插入排序,交换不相邻地元素以对数组地局部进行排序,最终用插入排序将局部有序的数组排序。
希尔排序的思想是使数组中任意间隔为 h 的元素都是有序的。这样的数组成为 h 有序数组。换句话说,一个 h 有序数组就是 h 个相互独立的有序数组组合在一起的一个数组。
在进行排序时,刚开始 h 很大,就能将元素移动到很远的地方,为实现更小的 h 有序创造方便。h 递减到 1 时,相当于插入排序。这就是希尔排序。下面的代码 h 从 N/3 开始递减,h = h * 3 + 1:
public class Shell: BaseSort
{
public static long usedTimes = 0;
public Shell()
{
}
public static void Sort(IComparable[] a)
{
Stopwatch timer = new Stopwatch();
timer.Start();
int n = a.Length;
int h = 1;
while (h < n / 3)
h = h * 3 + 1;
/*
* 每次循环生成 h有序数组(h 个有序数组) 即 (h 0), (h+1 1), (h+2 2), ....
* 如果右边的值小于左边的值时,交换;
* 并退回到左边的位置,重新排序,循环比较(j>=h)。
* 因为交换可能引起原有序的数组乱序,所以需要退回去重新比较
* 当 h 递减到 1时,相当于插入排序一个部分有序数组
* */
while (h >= 1)
{
//Console.WriteLine(h);
for (int i = h; i < n; i++)
{
//Console.WriteLine(i+","+(i - h));
for (int j = i; j >= h && Less(a[j], a[j - h]); j -= h)
{
Exch(a,j,j-h);
//Console.WriteLine("J:" + (j- h));
}
}
h = h / 3;
}
timer.Stop();
usedTimes = timer.ElapsedMilliseconds;
}
}
从另一个角度看希尔排序,插入排序是将每个比元素大的元素向右移动一个,希尔排序是交换到比它大的元素之前,移动距离从 1 改成 h 。
希尔排序更高效的原因是它将数组变成多个部分有序的数组,减少倒置。排序之初,各个子数组都很短,排序之后子数组都是部分有序的,这很适合插入排序。子数组部分有序的程度取决于递增序列的选择。(如何选择??)
和选择排序以及插入排序对比,希尔排序也可以用于大型数组,即使是随机数字排序也快很多。数组规模越大,对比越明显。
希尔排序的性能分析比较困难。它的运行时间达不到平方级别,在最坏情况下的比较次数和 N ^ 3/2 成正比。
如果中等大小的数组可以使用希尔排序,它代码量少,且不需要额外的内存空间。
百万级别的排序时间不超过十秒:
count:10000 shell use Milliseconds:6
count:100000 shell use Milliseconds:180
count:1000000 shell use Milliseconds:3226
来源:https://www.cnblogs.com/afei-24/p/13338901.html
猜你喜欢
- 一、什么是ASMASM是一个java字节码操纵框架,它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 class 文件,
- 1. 使用Files.list()迭代目录及其子目录文件Files.list()可以迭代目录及其子目录文件Files.list(Paths.
- (注意:本文基于JDK1.8)前言元素在存储到内存中,当我们需要使用在内存中存储的元素,这就涉及到在内存中查找元素,今天一起学习Vector
- Mybatis的Dao层实现传统开发方式编写UserDao接口public interface UserDao {  
- 前言:线程池是一个非常重要的知识点,也是池化技术的一个典型应用,相信很多人都有使用线程池的经历,但是对于线程池的实现原理大家都了解吗?本篇文
- Q1: Object类型包含哪些方法?A1: Object类型共包含6个方法,Equals, GetHashCode, ToString,
- Spring Cloud Gateway使用Spring Cloud Gateway是一个基于Spring Boot 2.x和Spring&
- 拿到了项目框架工程代码却没有uml图,那么方法之间的调用关系功能流转就不容易看出来,那么如何产生类图呢,记忆里方法有下:1.rose逆向工程
- Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。一、当两个并发线程访问同一个
- classProgram{ staticvoid Main() {&
- 大家可以自行百度下阿里分布式事务,在这里我就不啰嗦了。下面是阿里分布式事务开源框架的一些资料,本文是springboot+dubbo+fes
- 前言在实际生活中,地图是我们经常使用的一种工具,通常我们会用它进行导航,输入一个出发城市,输入一个目的地城市,就可以把路线规划好,而在规划好
- 前言HTML5 WebSocket实现了服务器与浏览器的双向通讯,双向通讯使服务器消息推送开发更加简单,最常见的就是即时通讯和对信息实时性要
- 1.功能介绍Spring框架提供了线程池和定时任务执行的抽象接口:TaskExecutor和TaskScheduler来支持异步执行任务和定
- 之前介绍了SpringBoot集成Jpa的简单使用,接下来介绍一下使用Jpa连接数据库对数据进行排序、分页、条件查询和过滤操作。首先创建Sp
- 技术要点org.springframework.web.context.request.async.DeferredResult<T&
- springboot项目不配置数据源启动报错spring boot默认会加载org.springframework.boot.autocon
- SpringBoot默认的存放静态资源文件的位置是:注:SpringBoot中的src/main/resources/资源文件夹对应clas
- 一、导入前言:导入必须用post请求具体原因在2中叙述1、Excel导入总结一下目标,就是要将excel中的数据行、逐一提取,最后得到一个l
- 本文实例为大家分享了java排列组合算法的具体代码,供大家参考,具体内容如下package BeanUtil;import java.uti