希尔排序又称为“缩小增量排序”,它属于插入类排序方法,是对直接插入排序的改进。
首先将待排序的元素分为多个子序列,使得每个子序列的元素个数相对较少,对各个子序列分别进行直接插入排序,待整个待排序序列“基本有序”后,再对所有元素进行一次直接插入排序。
以关键字序列{ 26 , 53 , 67 , 48, 57 , 13 , 48 , 32 , 60 , 50}为例,假设选择的步长序列为{5 , 3 , 1},则希尔排序的过程如图 9-2 所示。因为步长序列长度为 3,因此对待排序序列一共需要进行 3 趟排序。首先, 第一趟排序中将关键字序列分成 5 个子序列{26 , 13}, {53 , 48},{67 , 32}, {48 , 60}, {57 , 50},对它们分别进行直接插入排序,结果如图所示。然后,进行第二趟希尔排序,此时步长为 3,则将关键字序列分成 3 个子序列{13 , 48 , 53 , 57}, {48 ,50 , 67}, {32 , 26 , 60},对它们进行直接插入排序后的结果如图所示。最后,对整个序列进行一趟直接插入排序,此时得到一个关键字有序的序列,希尔排序结束。
在每趟排序过程中子序列的划分并不是简单的逐段划分,而是将间隔某个步长的元素组成一个子序列。如此,在对每个子序列进行简单插入排序时,关键字较小的元素就不是一步一步向前移动,而是按步长跳跃式向前移动,从而使得在进行最后一趟步长为 1 的插入排序时,整个序列已基本有序,此时,只需要作比较少的比较和移动即可完成排序。
import java.util.Arrays;
public class ShellSortTest {
public static void main(String[] args) {
int[] arr = {5,2,7,3,9,10,8,6,1,4};
shellSort(arr);
System.out.println(Arrays.toString(arr));
}
public static void shellSort(int[] arr){
int j;
int temp;
int gap = arr.length / 2;
while (gap > 0)
{
for (int i = gap; i < arr.length; i++)
{
j = i - gap;
temp = arr[i];
while ((j >= 0) && arr[j] > temp )
{
arr[j + gap] = arr[j];
j = j - gap;
}
arr[j + gap] = temp;
}
gap = gap / 2;
}
}
}
来一个整体的比较图吧~
小小总结一下