一篇讲透排序算法之插入排序and选择排序

1.插入排序

1.1算法思想

先将数组的第一个元素当作有序,让其后一个元素与其比较,如果比第一个元素小则互换位置,之后再将前两个元素当作有序的,让第三个元素与前两个元素倒着依次进行比较,如果第三个元素比第二个元素小的话,则交换位置,之后再比较第二个元素和第一个元素。

之后我们重复以上操作即可完成插入排序。

为了帮助大家理解,现在我们一步步完成这些操作。

1.2动图演示

1.3实现逻辑以及具体实现

首先,如果第二个元素小于第一个元素,则交换位置。

	if (a[1] < a[0])
	{
		int tmp = a[1];
		a[1] = a[0];
		a[0] = tmp;
	}

之后,我们就可以让第三个元素依次与前两个元素进行比较了。

在这里,我们要记录一下第二个元素的位置,否则我们不知道如何比较。

因此我们在这里定义一个end来记录第二个元素的位置。

int end = 1;
while (end)
{
    //如果第三个元素和第二个元素发生了交换
    //则比较第一个元素和新的第二个元素
	if (a[end + 1] < a[end])
	{
		int tmp = a[end + 1];
		a[end + 1] = a[end];
		a[end] = tmp;
	}
    //由于前end个元素已经有序,所以有一次没有进if就可以跳出循环了
    else
    {
        break;
    }
	end--;
}

现在我们就完成了单趟的循环,但我们需要比较多趟,而每趟的比较中不一样的变量就是end,每趟的比较end都会+1,因此我们在外层再写一个循环即可。

在这里我们再进行一些优化

//当i=n-2时,也就是前n-1个元素有序,此时第n个元素进入排序
//排序完了之后,整个数组有序,因此终止条件为i<n-1	
for (int i = 0; i < n-1; i++)
	{
		int end = i;
        int tmp = arr[i + 1];//保存插入元素
		while (end >= 0)
		{      //比较记录的tmp和end
               //每次都是tmp和新的end比较
			if (tmp < a[end])
			{ //由于我们记录了末尾后面一个的元素
              //这里直接把位置往后挪1即可
				a[end + 1] = a[end];
				end--;
			}
			else
			{
				break;
			}
		}
        //跳出循环后,更新end+1处数值
		a[end + 1] = tmp;
	}

到这里我们已经完成了一次插入排序,最后我们只需要将这些内容放在函数定义当中即可彻底完成!

void InsertSort(int* a, int n)
{
	for (int i = 0; i < n-1; i++)
	{
		int end = i;
        int tmp = arr[i + 1];//保存插入元素
		while (end >= 0)
		{      //比较记录的tmp和end
               //每次都是tmp和新的end比较
			if (tmp < a[end])
			{ //由于我们记录了末尾后面一个的元素
              //这里直接把位置往后挪1即可
				a[end + 1] = a[end];
				end--;
			}
			else
			{
				break;
			}
		}
        //跳出循环后,更新end+1处数值
		a[end + 1] = tmp;
	}
}

2.选择排序

2.1算法思想

先遍历一遍数组,找出最小的元素,然后与第一个元素交换位置。

之后从数组的第二个数开始,再遍历一遍数组,找出此次遍历中最小的数,与第二个数交换位置。

之后重复以上的操作即可完成任务。

使用上述的原理遍历数组,每次遍历都可以选出一个最小的数,放到对应的位置上面去。

2.2动图演示

2.3实现逻辑以及具体实现

现在我们来逐步实现这个排序算法

首先,我们应遍历数组,找出最小的元素。

int min=a[0];
for (int i = 0; i < n; i++)
{
	if (a[i] < min)
	{
		int tmp = min;
		min = a[i];
		a[i] = tmp;
	}
}

我们这样写出现了一个问题,我们交换了min和a[i]的数值,但是a[0]处的数值并没有随着min的改变而改变,对此,我们应该怎么做呢?

这里我们可以将两数交换封装为一个函数,之后我们不再让min记录数值,而是记录下标即可。 

//假设最小的是下标为0处的元素
int min = 0;
for (int i = 0; i < n; i++)
{

	if (a[i] < a[min])
	{
		//交换下标,每次交换后,min中记录的则是较小数的下标
		//之后进入下一次循环,继续比较
		min = i;
	}
}
//比较完毕之后,交换数值即可
swap(&arr[min], &arr[0]);//这个函数自己写,我没贴上来。

现在我们已经找到了最小的数,而且已经将其放到了第一个位置上去。

现在我们只需要在外面再套一层循环,每次不断更新下标,即可完成选择排序。

for (int i = 0; i <n-1; i++)//一共需要比较n-1趟
{
	//此时i下标元素最小
	int min = i;
	for (int j = i+1; j < n; j++)//前i个有序,因此j=i+1,比较n个元素,因此j<n
	{
		//假设最小的是下标为j处的元素
		if (a[j] < a[min])
		{
			//交换下标,每次交换后,min中记录的则是较小数的下标
			//之后进入下一次循环,继续比较
			min = j;
		}
	}
	//比较完毕之后,交换数值即可
	swap(&arr[min], &arr[0]);
}

 我们将其封装在函数体内,即可完成一次排序。

void SelectSort(int* arr, int len)
{
	for (int i = 0; i < len - 1; i++)
	{
		int min = i;
		for (int j = i + 1; j < len; j++)
		{
			if (arr[j] < arr[min])
			{
				mini = j;
			}
		}
		swap(&arr[min], &arr[i]);
	}
}

2.4选择排序的优化

我们的选择排序还可以进行一下优化,我们可以在一次循环中同时找最小和最大的数据,怎么样做呢?

只需要在每次比较时定义一个min,一个max,即可在一次循环中找到最小和最大的数据。

这里我们还是先完成第一趟排序,代码如下

int max = 0;
int min = 0;
for (int j =1; j < n; j++)//前i个有序,因此j=i+1,比较n个元素,因此j《n
{
	//假设最小的是下标为j处的元素
	if (a[j] < a[min])
	{
		//交换下标,每次交换后,min中记录的则是较小数的下标
		min = j;
	}
	if (a[j] > a[max])
	{
        //交换下标,每次交换后,min中记录的则是较大数的下标
		max = j;
	}
}
swap(&arr[min], &arr[0]);
swap(&arr[max], &arr[n-1]);

通过上面这段代码,我相信大家可以了解优化的原理了。

现在我们就可以着手于完成优化后的选择排序了。

由于我们在一个循环体内同时寻找最小值和最大值,并交换位置。

因此我们需要两个变量记录每次循环中最左边的值和最右边的值的下标,我们将其定义为begin和end。

当begin和end相遇时,我们的循环就结束了。这时我们的数组就有序了。

当begin>=end时,她们就有序了。因此我们的循环条件可以写为begin<end。

void SelectSort(int* arr, int len)
{
	int begin = 0;
	int end = len - 1;
	while (begin < end)
	{
		//假设mini和maxi都是begin
		int mini = begin;
		int maxi = begin;
		for (int i = begin + 1; i <= end; i++)
		{
			//找大
			if (arr[mini] > arr[i])
			{
				mini = i;
			}
			//找小
			if (arr[maxi] < arr[i])
			{
				maxi = i;
			}
		}
		swap(&arr[mini], &arr[begin]);
		swap(&arr[maxi], &arr[end]);
		begin++;
		end--;
		}
	}
}

到了这里,我们的选择排序已经基本完成了,

但是请大家思考一个问题: 

如果我们的begin和maxi重合的话,上面这段代码的执行逻辑是什么样的呢?

    如果begin和maxi重合,根据我们的代码逻辑,我们首先会交换mini和begin位置处的值。

    此时我们的begin下标处的值已经更新成了mini处的值,而由于我们的maxi和begin是相同的,因此此时第二次会将最大值和最小值交换,从而导致出现bug。

那么应该怎么解决这个问题呢?很简单,因为我们的下标重合了,那么我们在进行完第一个交换之后更新一下maxi的值即可完成交换。

void SelectSort(int* arr, int len)
{
	int begin = 0;
	int end = len - 1;
	while (begin < end)
	{
		//假设mini和maxi都是begin
		int mini = begin;
		int maxi = begin;
		for (int i = begin + 1; i <= end; i++)
		{
			//找大
			if (arr[mini] > arr[i])
			{
				mini = i;
			}
			//找小
			if (arr[maxi] < arr[i])
			{
				maxi = i;
			}
		}
		swap(&arr[mini], &arr[begin]);
		//begin处的值和mini处的值交换了位置
		//因此现在mini处保存的值为最大值
		//我们将maxi更新为mini即可。
		if (begin == maxi)
		{
			maxi = mini;
		}
		swap(&arr[maxi], &arr[end]);
		begin++;
		end--;
		}
	}
}

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值