排序-----插入排序,希尔排序

文章详细介绍了插入排序的基本思想,包括其将元素插入有序列表的过程,以及其时间复杂度和稳定性。接着讨论了希尔排序,这是一种对插入排序的优化,通过分组和增量逐步减少的方式提高排序效率。希尔排序在接近有序的情况下能更快完成,但其时间复杂度因增量序列的不同而变化。
摘要由CSDN通过智能技术生成

插入排序,希尔排序

插入排序

基本思想

插入排序是一种最简单的的排序思想,它的思想是将一个数据插入到一个有序的数据列表,得到一个新的有序列表。

把n个待排序的元素看成为一个有序表和一个无序表,开始时有序表中只包含一个元素,无序表中包含有n-1个元素,排序过程中每次从无序表中取出第一个元素,在有序表中从后往前进行比较,将它插入到有序表中的适当位置,使之成为新的有序表。

实际中我们玩扑克牌时,就用了插入排序的思想.
Alt

思维导图

在这里插入图片描述

特性总结

时间复杂度:O(N^2)

最好:O(N) ------顺序有序或者接近有序

最坏:O(N^2) ------逆序

空间复杂度:O(1)

稳定性:稳定

代码实现

void insertsort(int* a, int n)
{
	for (int i = 0; i < n-1; i++)
	{
		//n-1是数组最后一个元素的下标
		//end的取值范围是[0,n-2]
        //x是end的后一个元素,若end取n-1,那x就越界了
		int end = i;
		int x = a[end + 1];
        
		//单趟排序
		//将x插入到[0,end]的区间内
		while (end >= 0)
		{
			if (a[end] > x)
			{
				//a[end]>x ,a[end]往后移动一个位置
				//虽然会覆盖掉end+1的值,但是不影响
				//end+1的值已经被保存在x中
				a[end + 1] = a[end];
				end--;
			}
			else
			{
				break;
			}
		}
		a[end + 1] = x;
        
	}
}

希尔排序

基本思想

希尔排序是把元素按下标的一定增量进行分组,对每组使用直接插入排序算法排序。随着增量逐渐减少,当增量减至 1 时,整个文件恰被分成一组,算法便终止。

在希尔排序的内部使用的直接插入排序,通过使用直接插入排序使得每一个分组都能保持有序,当增量减为1时,那么整个序列为一个分组,因为一个分组时有序的,也就实现了完全的排序。

思维导图

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

特性总结

  1. 希尔排序是对直接插入排序的优化。

  2. **当gap > 1时都是预排序,目的是让数组更接近于有序。**当gap == 1时,数组已经接近有序的了,这样就会很快。这样整体而言,可以达到优化的效果。我们实现后可以进行性能测试的对比。

  3. 希尔排序的时间复杂度不好计算,因为gap的取值方法很多,导致很难去计算,因此在好些树中给出的希尔排序的时间复杂度都不固定。

  4. 时间复杂度:O(N^1.25) ~ O(1.6*N^1.25) -----《数据结构-用面相对象方法与C++描述》— 殷人昆

  5. 稳定性:不稳定

代码实现

void shellsort(int* a, int n)
{
	int gap = n;
    //gap为分组间隔
    //多趟排序(gap>1)+插入排序(gap==1)
	while (gap > 1)
	{
		gap = gap / 3 + 1;
        
        //多组一起排序
		for (int i = 0; i < n - gap; i++)
		{
            
            //单组单趟插入排序
			int end = i;
			int x = a[end + gap];
			while (end >= 0)
			{
				if (a[end] > x)
				{
					a[end + gap] = a[end];
					end-=gap;
				}
				else
				{
					break;
				}
			}
			a[end + gap] = x;
		}
        
	}
}

总结

插入排序的思想比较好理解,希尔排序是建立在插入排序上的优于插入排序的算法。希尔排序分为预排序和插入排序两步,当插入排序的数组接近有序时,时间复杂度为O(N),预排序的目的是将一个无序数组变得更加有序,这样希尔排序的时间复杂度更接近于O(N),但是希尔排序的时间复杂度并不好算,因为gap时变化的。

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
好的,我可以为你解答关于希尔排序的问题。希尔排序插入排序的一种改进算法,也被称为“缩小增量排序”。它的基本思想是将待排序的序列按照一定的间隔分成若干个子序列,对每个子序列进行插入排序,然后逐步缩小间隔,直到间隔为1,最后对整个序列进行一次插入排序希尔排序的时间复杂度与间隔序列的选择有关,最坏时间复杂度为$O(n^2)$,平均时间复杂度为$O(n \log n)$。相比于插入排序希尔排序的时间复杂度有了明显的提升,且其在实际应用中也表现得比插入排序好。 下面是希尔排序的示例代码: ```python def shell_sort(arr): n = len(arr) gap = n // 2 while gap > 0: for i in range(gap, n): temp = arr[i] j = i while j >= gap and arr[j - gap] > temp: arr[j] = arr[j - gap] j -= gap arr[j] = temp gap //= 2 return arr ``` 在这段代码中,我们首先定义了一个希尔排序的函数shell_sort,它接收一个列表arr作为参数,并返回排好序的列表。在函数中,我们首先获取列表的长度n和初始的间隔gap,然后开始循环,直到间隔为1为止。在每次循环中,我们将列表按照间隔分成若干个子序列,对每个子序列进行插入排序,将子序列按照升序排列。最后,我们将间隔除以2,继续循环直到间隔为1。 使用希尔排序对列表进行排序非常简单,只需要调用shell_sort函数即可。例如,我们可以这样调用函数: ```python arr = [5, 2, 8, 3, 1, 6] sorted_arr = shell_sort(arr) print(sorted_arr) ``` 这段代码会输出排好序的列表[1, 2, 3, 5, 6, 8]。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值