图解希尔排序---插入排序的优化(shell)

希尔排序(Shell Sort)也称缩小增量排序,类型是插入排序:时间复杂度为:O(n^(1.3-2))

个人理解,希尔排序是一个基于插入排序的优化排序法;

我们对长度为16目标数n组arr元素: 6  38  5  9  3  2  1  8  6  3  11  10  22  39    进行排序。

希尔排序步骤如下:

1、我们将数组分为两段,很每段第X个元素角标差值为:gap = arr.length/2; 我们称为步长。

图解希尔排序---插入排序的优化(shell)_第1张图片

 2、对arr[i]和arr[i+gap]进行比较排序

图解希尔排序---插入排序的优化(shell)_第2张图片

 3、第一次排序之后的数组变为:

 4、接下来再将数组分为:4段,步长变为gap = gap / 2 = 4;

 5、此时一次比较4个元素:

图解希尔排序---插入排序的优化(shell)_第3张图片

6、第二次排序后数组变为:

 7、接下来再将数组分为:8段,步长变为gap = gap / 2 = 2,此时一次比较8个元素。

图解希尔排序---插入排序的优化(shell)_第4张图片

8、此时数组变为:

 可以发现此时数组已经变得越来越有序了,这样就减少了最后一步排序的负担:

9、最后就是gap = 1,对所有元素进行顺序排序,此时对于这样一个大部分都已经顺序的数组来说,是很好排的。这就是优化;

 

从分析图可知,无论如何,希尔排序都要经过最后一个步长为1的排序,也就是和我们直接排序一个一个排好像没什么区别?但是它优化的地方在于当进行最后一次排序的时候,它的复杂度不高,它此时已经是大部分有序的了,所以排起来也是很好排的。

希尔排序的基本思想:设待排序元素序列有n个元素,首先取一个整数increment(小于n)作为间隔将全部元素分为increment个子序列,所有距离为increment的元素放在同一个子序列中,在每一个子序列中分别实行直接插入排序。然后缩小间隔increment,重复上述子序列划分和排序工作。直到最后取increment=1,将所有元素放在同一个子序列中排序为止。

代码实现:

public class Shell{
	public static void main(String[] args) {
		int [] arr = {7,9,6,38,5,9,3,2,1,8,6,3,11,10,22,39};
		insert_sort(arr);
		for (int i : arr) {
			System.out.print(i + " ");
		}
	}
	public static void insert_sort(int[] arr) {
		
		int len = arr.length;
		int gap = len/2;
		
		while(gap > 0) {
			
			for (int i = gap; i < arr.length; i++) {
				int j = i;int e = arr[i];
				while (j-gap >= 0 && e < arr[j-gap]) {
					arr[j] = arr[j-gap];
					j -= gap;
				}
				arr[j] = e;
				
			}
			gap /= 2;
		}
	}
}

总结:

专家们提倡,几乎任何排序工作在开始时都可以用希尔排序,若在实际使用中证明它不够快,再改成快速排序这样更高级的排序算法.本质上讲,希尔排序算法是直接插入排序算法的一种改进,减少了其复制的次数,速度要快很多。

假如一个很小的数据在靠右端的位置上,那么要将该数据排序到正确的位置上,则所有的中间数据都需要向右移动一位。

希尔排序通过加大插入排序中元素之间的间隔,并对这些间隔的元素插入排序,从而使得数据可以大幅度的移动。当完成该间隔的排序后,希尔排序会减少数据的间隔在进行排序。依次进行下去。

希尔排序是按照不同步长对元素进行插入排序,当刚开始元素很无序的时候,步长最大,所以插入排序的元素个数很少,速度很快;当元素基本有序了,步长很小,插入排序对于有序的序列效率很高。所以,希尔排序的时间复杂度会比o(n^2)好一些。

希尔排序就是对直接插入排序的一种优化过程!

 

 

 

 

 

 

 

 

 

 

 

你可能感兴趣的:(《曾杂》,Java学习,希尔排序,图解,算法)