数据结构——直接插入排序与希尔排序(图示+文字详解)

内容包括:排序的代码实现,排序原理详解,代码详解,图示

part 1:直接插入排序

代码实现:

void InsertSort(int* a, int n)
{
	int i = 0;
	for (i = 0; i < n - 1; i++)
	{
		int end = i;
		int tmp = a[i + 1];
		while (end >= 0)
		{
			if (a[end] > tmp)
			{
				a[end + 1] = a[end];
				end--;
			}
			else
			{
				break;
			}
		}
		a[end + 1] = tmp;
	}
}

效果图例:

 

 排序原理:

基本原理:向一个有序区间插入一个值(此值在有序区间末尾的后一个位置)

生动些:可以将这种插入排序看作摸牌的过程,我们手上的牌都是已经按照一定顺序排列好的,现在我们又新拿了一张牌,准备插入,方法就是从手上排好的最后一张牌开始,依次比较

若是当前手上的牌大于待插入的新牌,那么需要将手上的此牌向后移动一步,腾出一个位置

若是当前手上的牌小于待插入的新牌,则新牌的位置则定下,即手上的此牌的后一个位置

代码详解:

void InsertSort(int* a, int n)
{
	int i = 0;
	for (i = 0; i < n - 1; i++)//所有有序区间的末尾位置
	{
		int end = i;//有序区间的末尾下标
		int tmp = a[i + 1];//待插入的元素先保存到tmp中
		while (end >= 0)//依次与有序区间中的所有元素一一比较
		{
			if (a[end] > tmp)
			{
				a[end + 1] = a[end];//有序区间的某个元素后移
				end--;//继续与有序区间中的元素比较
			}
			else
			{
				break;
			}
		}
		a[end + 1] = tmp;//待插入元素插入在有序区间某个元素的后面 or 越过整个有序区间,成为第一个
	}
}

1 for循环构造出所有有序区间末尾下标的可能

2 待插入元素需要临时保存在tmp变量中,因为在插入过程中伴随着有序区间元素的后移,若是不提前保存待插入元素的值,则可能导致待插入元素被覆盖

3 若是待插入元素比有序区间的元素小,则此有序区间的元素即其后面都需要后移一位,但是比较还在继续,直至待插入元素比有序区间的元素大,才会停止比较

4 停止比较的两种可能(待插入元素确定位置的两种情况):

                            1 待插入元素在有序区间内找到一个比小的元素,则它需要插在此元素的后面,即end+1的位置

                            2 待插入元素比所有有序区间内的元素都要小,此时end=-1,比无可比,则所有有序区间内的元素都后移一位,它插在数组的第一个位置上,即end+1的位置

故而在代码中我设计:若是遇到比待插入元素小的元素直接break,这样可以统一。

part 2:希尔排序

希尔排序是直接插入排序基础之上的代码设计

代码实现:

void ShellSort(int* a, int n)
{
	int gap = n;
	while (gap > 1)
	{
		gap = gap / 3 + 1;
		int i = 0;
		for (i = 0; i < n - gap; i++)
		{
			int end = i;
			int tmp = a[end + gap];
			while (end >= 0)
			{
				if (a[end] > tmp)
				{
					a[end + gap] = a[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}
			a[end + gap] = tmp;
		}
	}
}

排序原理:

基本原理:第一步:多次预排序使得序列逼近有序(gap>1)

                  第二步:直接插入排序,使得序列最终有序(gap==1)

间隔gap分为一组,共又gap组

gap=gap/3+1 或者gap=gap/2   无论哪种都会使得gap最终=1,就是所有元素构成一组,此时对一组的排序就是前面所介绍的直接插入排序,会让数组最终有序

代码详解:

void ShellSort(int* a, int n)
{
	int gap = n;
	while (gap > 1)//多次预排,gap=1时数组已经最终有序,结束循环
	{
		gap = gap / 3 + 1;//gap的变化,保证了gap的最后一次值是1
		int i = 0;
		for (i = 0; i < n - gap; i++)//有序区间末尾的下标,多组并排,i的变化即end的变化,end的值代表当前插入排序的是哪一组,end的取值必须<n-gap,下面有详解原因
		{
			int end = i;
			int tmp = a[end + gap];
			while (end >= 0)//对一组的某个待排序元素进行插入排序
			{
				if (a[end] > tmp)
				{
					a[end + gap] = a[end];
					end -= gap;//间隔gap为一组,所以end的前一个元素是end-gap
				}
				else
				{
					break;
				}
			}
			a[end + gap] = tmp;
		}
	}
}

下面省略每组的每个待排序元素插入自己所在组有序区间的结果,读者可自行操作,这里介绍方法 

 

 end处在红组:对红组的一个待排序元素进行插入排序,初始有序区间内只有一个9

end++,end处在绿组,对绿组的一个待排序元素进行插入排序,初始有序区间内只有一个8

end++,end处在蓝组, 对蓝组的一个待排序元素进行插入排序,初始有序区间内只有一个7

 

end++,end处在橙组, 对橙组的一个待排序元素进行插入排序,初始有序区间内只有一个6

之后的end继续++,先是将红组中的下一个待插入元素进行插入排序,再是对绿组的下一个待插入元素进行插入排序,然后是对蓝组的下一个待插入元素进行插入排序,再是对橙组的下一个待插入元素进行插入排序,红,绿,蓝,橙……如此循环多组并排

但是end最终只能抵达n-gap-1的位置,即,end<n-gap,因为我们的操作是将end+gap位置的待插入元素进行插入,必须保持数组的下标不越界,当end处在n-gap位置时,它的后一个元素end+gap就是n了,属于非法下标 ,而且当end抵达最终位置时,这一趟的预排序已经处理完成

 完结撒花~

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值