Java数据结构之排序---希尔排序
希尔排序的基本介绍:
希尔排序同之前的插入排序一样,它也是一种插入排序,只不过它是简单插入排序之后的一个优化的排序算法,希尔排序也被称为缩小增量排序。
希尔排序的基本思想:
希尔排序是把数组中给定的元素按照下标的一定增量进行分组,在分组之后,对每组使用直接插入排序算法;随着增量的减少,每组包含的元素越来越多,当增量减少到1的时候,整个数组正好被分成一组,此时该算法终止。通常我们判断增量是通过:第一次的增量=数组的长度/2(取整),第二次的增量=第一次的增量/2(取整)...一直到增量为1结束。
希尔排序的示意图:
整个希尔排序我们可以通过两种方式来实现:交换法(用交换排序的思想),移动法(用插入排序的思想)。在这个例子中,我们给定的数组是int[] arr = {8,9,1,7,2,3,5,4,6,0};
(1).交换法
其中交换法是通过两种方式讲述的:分步骤的实现,整体的实现。具体的解释在代码的注释中,观看注释即可。
(1.1)分步骤的实现
public static void main(String[] args) {
// TODO Auto-generated method stub
int[] arr = {8,9,1,7,2,3,5,4,6,0};
System.out.println("原始的序列:");
System.out.println(Arrays.toString(arr));
shellSort(arr);
} //交换法
//希尔排序
public static void shellSort(int[] arr){ //第一趟排序,由于我们数组中打的元素一共有10个,所以我们第一趟选择的增量为10/2=5
int temp = 0;
for(int i=5;i<arr.length;i++){ //我们这里令i从5开始
for(int j=i-5;j>=0;j-=5){ //j代表了与i对应在一个数组中的数。
if(arr[j]>arr[j+5]){ //这里面就是用来比较在同一个数组里面的数字的大小,并且为它们排序
temp = arr[j];
arr[j] = arr[j+5];
arr[j+5] = temp;
}
}
}
System.out.println("第一趟结束后的序列:");
System.out.println(Arrays.toString(arr)); //第二趟序列,这里面的增量是第一趟的增量/2=5/2=2
for(int i=2;i<arr.length;i++){
for(int j=i-2;j>=0;j=j-2){ //同理,跟第一趟相同
if(arr[j]>arr[j+2]){
temp = arr[j];
arr[j] = arr[j+2];
arr[j+2] = temp;
}
}
}
System.out.println("第一趟结束后的序列:");
System.out.println(Arrays.toString(arr)); //第三趟序列,这里面的增量是第二趟的增量/2=2/2=1
for(int i=1;i<arr.length;i++){
for(int j=i-1;j>=0;j=j-1){ //同理,跟第一趟相同
if(arr[j]>arr[j+1]){
temp = arr[j];
arr[j] = arr[j+1];
arr[j+1] = temp;
}
}
}
System.out.println("第一趟结束后的序列:");
System.out.println(Arrays.toString(arr));
}
上述代码的最终结果如下:
(1.2)完整的代码
由上述代码,我们可以看到,当我们增量发生改变的时候,我们只是需要改变j的起始值以及定长的值即可,因此我们可以写出完整的代码如下:
public static void main(String[] args) {
// TODO Auto-generated method stub
int[] arr = {8,9,1,7,2,3,5,4,6,0};
System.out.println("原始的序列:");
System.out.println(Arrays.toString(arr));
shellSort(arr);
} //交换法
//希尔排序
public static void shellSort(int[] arr){ //完整的希尔排序算法
int temp = 0;
int count = 0;
//gap的值代表了就是每一趟的增量的值
for(int gap = arr.length/2;gap>=1;gap = gap/2){
for(int i=gap;i<arr.length;i++){ //如之前的代码,i从增量的位置开始算起
for(int j=i-gap;j>=0;j=j-gap){ //这里面同之前的代码,只不过把增量变成了gap
if(arr[j]>arr[j+gap]){
temp = arr[j];
arr[j] = arr[j+gap];
arr[j+gap] = temp;
}
}
}
System.out.println("第"+(++count)+"趟排序后的序列是:");
System.out.println(Arrays.toString(arr));
}
}
(2).移动法
上述代码是希尔排序的选择类的写法,它由一定的缺陷,就是效率的问题,我们每一趟都需要进行比较,这样会使整个代码的时间复杂度增加。因此我们用移动法对其进行优化。代码如下:
public static void main(String[] args) {
// TODO Auto-generated method stub
int[] arr = {8,9,1,7,2,3,5,4,6,0};
System.out.println("原始的序列:");
System.out.println(Arrays.toString(arr));
shellSort2(arr);
}
//对交换式的希尔排序进行优化,移动法
public static void shellSort2(int[] arr){
int count = 0;
for(int gap = arr.length/2;gap>=1;gap = gap/2){ //这里同上述代码保持不变
//下面的代码根据我们的插入算法一样,只不过之前的插入算法的增量一直是1,这里面的增量是变化的,具体参照之前一章---插入算法
for(int i=gap;i<arr.length;i++){
int minIndex = i-gap; //待插入位置的下标
int temp = arr[i]; //要插入的数
while(minIndex>=0 && temp<arr[minIndex]){
arr[minIndex+gap] = arr[minIndex];
minIndex-=gap;
}
arr[minIndex+gap] = temp;
}
System.out.println("第"+( ++count)+"趟排序的结果:");
System.out.println(Arrays.toString(arr));
}
}
上述代码的最终结果如下:
Java数据结构之排序---希尔排序的更多相关文章
- java数据结构和算法------希尔排序
package iYou.neugle.sort; public class Shell_sort { public static void ShellSort(double[] array) { i ...
- Java数据结构和算法 - 高级排序
希尔排序 Q: 什么是希尔排序? A: 希尔排序因计算机科学家Donald L.Shell而得名,他在1959年发现了希尔排序算法. A: 希尔排序基于插入排序,但是增加了一个新的特性,大大地提高了插 ...
- C语言实现 冒泡排序 选择排序 希尔排序
// 冒泡排序 // 选择排序 // 希尔排序 // 快速排序 // 递归排序 // 堆排序 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h& ...
- 数据结构复习:希尔排序的C++实现
1.原理介绍 希尔排序又称为缩小增量排序,由D.L.Shell在1959年提出而得名. 该算法先取一个小于数据表中元素个数 n 的整数gap, 并以此作为第一个间隔,将数据分为gap个子序列,所有距离 ...
- Java数据结构和算法 - 简单排序
Q: 冒泡排序? A: 1) 比较相邻的元素.如果第一个比第二个大,就交换它们两个; 2) 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对.在这一点,最后的元素应该会是最大的数; 3) 针 ...
- 排序---希尔排序Java
希尔排序 插入排序的一种又称“缩小增量排序”,是直接插入排序算法的一种更高效的改进版本.希尔排序是非稳定排序算法. 希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序:随着增量逐渐减 ...
- java数据结构1--数组、排序和Arrays工具类
数组:Array 数组的定义 数组的内存结构 数组定义常见问题 数组常见操作 Java参数传递问题--值传递 二维数组 1.数组概念 同一种类型数据的集合,可以是基本数据类型,也可以是引用数据类型. ...
- Java数据结构(七)—— 排序算法
排序算法(Sort Algorithm) 排序算法介绍和分类 将一组数据,依指定顺序进行排列 排序的分类 内部排序 指将需要处理的所有数据都加载到内部存储器中进行排序 外部排序 数据量过大,无法全部加 ...
- Java数据结构与算法之排序
排序从大体上来讲,做了两件事情: 1.比較两个数据项: 2.交换两个数据项.或复制当中一项 一.冒泡排序 大O表示法:交换次数和比較次数都为O(N*N). 算法原理: 1.比較相邻的元素.假设第一个比 ...
随机推荐
- Statistics项目学习笔记
1. http://218.244.157.0:55443/index.html 初始访问时,弹出的窗口为index.html文件,文件有html命令组成.html展现的UI界面用的是WIN10-UI ...
- JVM 堆内存设置原理(转)
堆内存设置 原理 JVM堆内存分为2块:Permanent Space 和 Heap Space. Permanent 即 持久代(Permanent Generation),主要存放的是Java类定 ...
- Tomcat 调优的技巧 (转)
描述 最近在补充自己的短板,刚好整理到Tomcat调优这块,基本上面试必问,于是就花了点时间去搜集一下tomcat调优都调了些什么,先记录一下调优手段,更多详细的原理和实现以后用到时候再来补充记录,下 ...
- python2.7.5升级到2.7.14或者直接升级到3.6.4
python2.7.5升级到2.7.14 1.安装升级GCC yum install -y gcc* openssl openssl-devel ncurses-devel.x86_64 bzip2 ...
- HRBUST 1849 商品中心
vjudge 智商掉线... 可以发现一条边能贡献其他点当且仅当两点路径上这个边权值最小,所以如果按照边权从大到小加边,每加一条边就会合并两个联通块,那么一个联通块内的点到另一个联通块的点的权值就都是 ...
- 使用html2canvas实现屏幕截图
相关文件(vue3.0) <script src="https://cdn.jsdelivr.net/bluebird/latest/bluebird.js">< ...
- ld - GNU linker (连接器)
总览 (SYNOPSIS) ld [-o output] objfile... [-Aarchitecture] [-b input-format] [-Bstatic] [-Bdynamic] [- ...
- Matlab 中 Data-driven 风格的 API 设计
设计 所谓 data-driven API,指的是用户可以把"操作"作为参数,传入函数,像下面这种: stream = dataStream('load', 'example.cs ...
- 一、Core基于MVC的全局过滤器验证
一.Core基于MVC的过滤器验证 1.添加一个过滤器.在Startup 中ConfigureServices方法里添加一个Filters 即我们自己授权代码类. public void Config ...
- spring之bean的自动扫描
首先看一段applicationContext.xml中的自动扫描配置 <context:component-scan base-package="org.java.test" ...