希尔排序
希尔排序是插入排序的一种,又称“缩小增量排序”,是插入排序算法的一种更高效的改进版本。
引入希尔排序的原因是插入排序在某些时刻看起来并不高效,例如如果已排序的分组元素为{2,5,7,9,10},未排序的分组元素为{1,8},那么下一个待插入元素为1,我们需要拿着1从后往前,依次和10,9,7,5,2进行交换位置,才能完成真正的插入,每次交换只能和相邻的元素交换位置。那如果我们要提高效率,直观的想法就是一次交换,能把1放到更前面的位置,比如一次交换就能把1插到2和5之间,这样一次交换1就向前走了5个位置,可以减少交换的次数,这时候就引入了希尔排序。
排序原理:
- 选定一个增长量h,按照增长量h作为数据分组的依据,对数据进行分组;
- 对分好组的每一组数据完成插入排序;
- 减小增长量,最小减为1,重复第二步操作。
public static void main(String[] args) {
int arr[]={4,2,1,5,9,7,8,6};
int h=1;
//设定步长的初始长度
while (h<arr.length){
h= 2*h+1;
}
while (h>=1){
// 找到每组待插入的元素中的元素
for (int i=h; i<arr.length; i++){
// 找到每组已经排序好的元素进行比较并插入
for (int j=i; j>=h; j-=h){
if (arr[j]<arr[j-h]){
int temp;
temp=arr[j];
arr[j]=arr[j-h];
arr[j-h]=temp;
}else {
break;
}
}
}
h=h/2;
}
System.out.println(Arrays.toString(arr));
}
时间复杂度:在希尔排序中,增长量h并没有固定的规则,有很多论文研究了各种不同的递增序列,但都无法证明某个序列是最好的,对于希尔排序的时间复杂度分析,没法直接认定。但通过实例与插入排序进行比较的话,性能有明显的提升。
总结
希尔排序是插入排序的一个变型,引入步长这个参数,将数组进行分组,以达到较好的性能。
实现时同样也使用两个for循环,分别是找到每组待排序的元素以及每组已经排序好的元素。