希尔排序的基本介绍:

希尔排序同之前的插入排序一样,它也是一种插入排序,只不过它是简单插入排序之后的一个优化的排序算法,希尔排序也被称为缩小增量排序。

希尔排序的基本思想:

希尔排序是把数组中给定的元素按照下标的一定增量进行分组,在分组之后,对每组使用直接插入排序算法;随着增量的减少,每组包含的元素越来越多,当增量减少到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数据结构之排序---希尔排序的更多相关文章

  1. java数据结构和算法------希尔排序

    package iYou.neugle.sort; public class Shell_sort { public static void ShellSort(double[] array) { i ...

  2. Java数据结构和算法 - 高级排序

    希尔排序 Q: 什么是希尔排序? A: 希尔排序因计算机科学家Donald L.Shell而得名,他在1959年发现了希尔排序算法. A: 希尔排序基于插入排序,但是增加了一个新的特性,大大地提高了插 ...

  3. C语言实现 冒泡排序 选择排序 希尔排序

    // 冒泡排序 // 选择排序 // 希尔排序 // 快速排序 // 递归排序 // 堆排序 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h& ...

  4. 数据结构复习:希尔排序的C++实现

    1.原理介绍 希尔排序又称为缩小增量排序,由D.L.Shell在1959年提出而得名. 该算法先取一个小于数据表中元素个数 n 的整数gap, 并以此作为第一个间隔,将数据分为gap个子序列,所有距离 ...

  5. Java数据结构和算法 - 简单排序

    Q: 冒泡排序? A: 1) 比较相邻的元素.如果第一个比第二个大,就交换它们两个; 2) 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对.在这一点,最后的元素应该会是最大的数; 3) 针 ...

  6. 排序---希尔排序Java

    希尔排序 插入排序的一种又称“缩小增量排序”,是直接插入排序算法的一种更高效的改进版本.希尔排序是非稳定排序算法. 希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序:随着增量逐渐减 ...

  7. java数据结构1--数组、排序和Arrays工具类

    数组:Array 数组的定义 数组的内存结构 数组定义常见问题 数组常见操作 Java参数传递问题--值传递 二维数组 1.数组概念 同一种类型数据的集合,可以是基本数据类型,也可以是引用数据类型. ...

  8. Java数据结构(七)—— 排序算法

    排序算法(Sort Algorithm) 排序算法介绍和分类 将一组数据,依指定顺序进行排列 排序的分类 内部排序 指将需要处理的所有数据都加载到内部存储器中进行排序 外部排序 数据量过大,无法全部加 ...

  9. Java数据结构与算法之排序

    排序从大体上来讲,做了两件事情: 1.比較两个数据项: 2.交换两个数据项.或复制当中一项 一.冒泡排序 大O表示法:交换次数和比較次数都为O(N*N). 算法原理: 1.比較相邻的元素.假设第一个比 ...

随机推荐

  1. Python time strptime()方法 时间操作

    描述 Python time strptime() 函数根据指定的格式把一个时间字符串解析为时间元组. 语法 strptime()方法语法: time.strptime(string[, format ...

  2. 4种vue当中的指令和它的用法

    1.v-if:判断是否隐藏 2.v-for:数据循环 3.v-bind:class:绑定一个属性 4.v-model:实现数据双向绑定 这里重点说明一个v-if和v-show的区别: 共同点:都是通过 ...

  3. vue-搜索功能-实时监听搜索框的输入,N毫秒请求一次数据

    <template> <div class="search-box"> <input class="box" :placehold ...

  4. help_topic表,以字符拆分,一行转多行

      help_topic表是数据库mysql下的一个表        SUBSTRING_INDEX(s, delimiter, number)        返回从字符串 s 的第 number 个 ...

  5. 【AST篇】教你如何编写 Eslint 插件

    前言 虽然现在已经有很多实用的 ESLint 插件了,但随着项目不断迭代发展,你可能会遇到已有 ESLint 插件不能满足现在团队开发的情况.这时候,你需要自己来创建一个 ESLint 插件. 本文我 ...

  6. iPhone屏幕适配,历史及现状(http://hjcapple.github.io/2014/10/10/iphone-screen.html)

    iPhone屏幕适配,历史及现状 初代iPhone 2007年,初代iPhone发布,屏幕的宽高是320×480像素.下文也是按照宽度,高度的顺序排列.这个分辨率一直到iPhone 3GS的也保持不变 ...

  7. java中too many characters in character literal

    IDE里发现 too many characters in character literal 翻译过来就是 字符中的字符过多 , 一般情况是:把 多个文字 放在了 '' (单引号)里,应该放到 &q ...

  8. zabbix 添加图行树

    1.安装graphtree cd /usr/share/zabbix wget https://raw.githubusercontent.com/OneOaaS/graphtrees/master/ ...

  9. C#.net中的rank方法

    string[,] abcd = new string[2, 4];abcd[0, 0] = "a";abcd[0, 1] = "b";abcd[0, 2] = ...

  10. Codeforces 989 P循环节01构造 ABCD连通块构造 思维对云遮月参考系坐标轴转换

    A 直接判存不存在连续的三个包含A,B,C就行 /*Huyyt*/ #include<bits/stdc++.h> #define mem(a,b) memset(a,b,sizeof(a ...