希尔排序

希尔排序Shell Sort是基于插入排序的一种改进

希尔排序的时间性能优于直接插入排序的原因:

当文件初态基本有序时直接插入排序所需的比较和移动次数均较少。

第一块希尔排序介绍

准备待排数组[6 2 4 1 5 9]

首先需要选取关键字,例如关键是3和1(第一步分成三组,第二步分成一组),那么待排数组分成了以下三个虚拟组:

[6 1]一组

[2 5]二组

[4 9]三组

看仔细啊,不是临近的两个数字分组,而是3(分成了三组)的倍数的数字分成了一组,

就是每隔3个数取一个,每隔三个再取一个,这样取出来的数字放到一组,

把它们当成一组,但不实际分组,只是当成一组来看,所以上边的"组"实际上并不存在,只是为了说明分组关系

对以上三组分别进行插入排序变成下边这样

[1 6] [2 5] [4 9]

具体过程:

[6 1]6和1交换变成[1 6]

[2 5]2与5不动还是[2 5]

[4 9]4与9不动还是[4 9]

第一趟排序状态演示:

待排数组:[6 2 4 1 5 9]

排后数组:[1 2 4 6 5 9]

第二趟关键字取的是1,即每隔一个取一个组成新数组,实际上就是只有一组啦,隔一取一就全部取出来了嘛

此时待排数组为:[1 2 4 6 5 9]

直接对它进行插入排序

还记得插入排序怎么排不?复习一下

[1 2 4]都不用动,过程省略,到5的时候,将5取出,在前边的有序数组里找到适合它的位置插入,就是4后边,6前边

后边的也不用改,所以排序完毕

顺序输出结果:[1 2 4 5 6 9]

第二块希尔排序的关键是如何取关键字,因为其它内容与插入排序一样

那么如何选取关键字呢?就是分成三组,一组,这个分组的依据是什么呢?为什么不是二组,六组或者其它组嘞?

好的增量序列的共同特征:

① 最后一个增量必须为1

② 应该尽量避免序列中的值(尤其是相邻的值)互为倍数的情况



  1. public class ShellSort {  
  2.     public static void shellSort(int[] data) {  
  3.         int j = 0;  
  4.         int temp = 0;  
  5.         for (int increment = data.length / 2; increment > 0; increment /= 2) {  
  6.             for (int i = increment; i < data.length; i++) {  
  7.                 temp = data[i];  
  8.                 for (j = i; j >= increment; j -= increment) {  
  9.                     if(temp > data[j - increment]){  
  10.                         data[j] = data[j - increment];  
  11.                     }else{  
  12.                         break;  
  13.                     }  
  14.                 }   
  15.                 data[j] = temp;  
  16.             }  
  17.         }  
  18.     }  
  19.   
  20.     public static void main(String[] args) {  
  21.         int[] data = new int[] { 528913 ,4};  
  22.   
  23.         System.out.println("未排序前");  
  24.         for (int i = 0; i < data.length; i++){  
  25.             System.out.print(data[i] + " ");  
  26.         }  
  27.               
  28.         shellSort(data);  
  29.           
  30.         System.out.println("\n排序后");  
  31.         for (int i = 0; i < data.length; i++)  
  32.             System.out.print(data[i] + " ");  
  33.     }  
  34.   
  35. }  

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值