c语言的5种常用排序算法

1、冒泡排序(最常用)
冒泡排序是最简单的排序方法:原理是:从左到右,相邻元素进行比较。每次比较一轮,就会找到序列中最大的一个或最小的一个。这个数就会从序列的最右边冒出来。(注意每一轮都是从a[0]开始比较的)

  • 以从小到大排序为例,第一轮比较后,所有数中最大的那个数就会浮到最右边;第二轮比较后,所有数中第二大的那个数就会浮到倒数第二个位置……就这样一轮一轮地比较,最后实现从小到大排序。
#include<stdio.h>
int main()
{
	int i,j,t,a[10];
	printf("输入10个整数:\n");
	for(i=0;i<10;i++)
		scanf("%d",&a[i]);
	for(i=0;i<9;i++)				//变量i仅代表比较的趟数(n-1)趟
		for(j=0;j<9-i;j++)
			if(a[j]>a[j+1])			//相邻两个数相互比较
			{
				t=a[j];
				a[j]=a[j+1];
				a[j+1]=t;
			}
	for(i=0;i<10;i++)
		printf("%5d",a[i]);
	printf("\n");
	return 0;
}

2、鸡尾酒排序
鸡尾酒排序又称双向冒泡排序、鸡尾酒搅拌排序、搅拌排序、涟漪排序、来回排序或快乐小时排序, 是冒泡排序的一种变形。该算法与冒泡排序的不同处在于排序时是以双向在序列中进行排序。
原理:数组中的数字本是无规律的排放,先找到最小的数字,把他放到第一位,然后找到最大的数字放到最后一位。然后再找到第二小的数字放到第二位,再找到第二大的数字放到倒数第二位。以此类推,直到完成排序。

#include<stdio.h>
int main()
{
	void cocktail_sort(int a[],int left,int right);
	int i,a[10];	
	printf("输入10个整数:\n");
	for(i=0;i<10;i++)
		scanf("%d",&a[i]);
	cocktail_sort(a,0,9);
	for(i=0;i<10;i++)
		printf("%5d",a[i]);	
	printf("\n");
	return 0;
}
void cocktail_sort(int a[],int left,int right)
{
	if(left>=right)
	{
		return;
	}
	int i,t;
	if(left<right)
	{
		for(i=left;i<right;i++)
			if(a[i]>a[i+1])//最大值移动到最右端处
			{
				t=a[i];
				a[i]=a[i+1];
				a[i+1]=t;
			}
		for(i=right;i>left;i--)
			if(a[i]<a[i-1])//最小值移动到最左端处
			{
				t=a[i];
				a[i]=a[i-1];
				a[i-1]=t;
			}
	cocktail_sort(a,left+1,right-1);	//重复运算,直到left>=right(此处使用递归调用)
	}	
}

3、快速排序(Quicksort)【是对冒泡排序的一种改进】
快速排序算法通过多次比较和交换来实现排序,其排序流程如下:
(1)首先设定一个分界值,通过该分界值将数组分成左右两部分。
(2)将大于或等于分界值的数据集中到数组右边,小于分界值的数据集中到数组的左边。此时,左边部分中各元素都小于或等于分界值,而右边部分中各元素都大于或等于分界值。
(3)然后,左边和右边的数据可以独立排序。对于左侧的数组数据,又可以取一个分界值,将该部分数据分成左右两部分,同样在左边放置较小值,右边放置较大值。右侧的数组数据也可以做类似处理。
(4)重复上述过程,可以看出,这是一个递归定义。通过递归将左侧部分排好序后,再递归排好右侧部分的顺序。当左、右两个部分各数据排序完成后,整个数组的排序也就完成了

原理
设要排序的数组是A[0]……A[N-1],首先任意选取一个数据(通常选
用数组的第一个数)作为关键数据,然后将所有比它小的数都放到它左边,所有比它大的数都放到它右边,这个过程称为一趟快速排序。值得注意的是,快速排序不是一种稳定的排序算法,也就是说,多个相同的值的相对位置也许会在算法结束时产生变动。
一趟快速排序的算法是:
1)设置两个变量i、j,排序开始的时候:i=0,j=N-1;
2)以第一个数组元素作为关键数据,赋值给key,即key=A[0];
3)从j开始向前搜索,即由后开始向前搜索(j–),找到第一个小于key的值A[j],将A[j]和A[i]的值交换;
4)从i开始向后搜索,即由前开始向后搜索(i++),找到第一个大于key的A[i],将A[i]和A[j]的值交换;
5)重复第3、4步,直到i=j; (3,4步中,没找到符合条件的值,即3中A[j]不小于key,4中A[i]不大于key的时候改变j、i的值,使得j=j-1,i=i+1,直至找到为止。找到符合条件的值,进行交换的时候i, j指针位置不变。另外,i==j这一过程一定正好是i+或j-完成的时候,此时令循环结束)。

#include<stdio.h>
int main()
{
	void quicksort(int a[],int left,int right);//函数声明
	int i,a[10];
	printf("输入10个整数:\n");
	for(i=0;i<10;i++)
		scanf("%d",&a[i]);
	quicksort(a,0,9);
	for(i=0;i<10;i++)
		printf("%5d",a[i]);
	printf("\n");
	return 0;
}
void quicksort(int a[],int left,int right)//函数实现快速排序
{
	if(left>=right)		/*如果左边索引大于或者等于右边的索引就代表已经整理完成一个组了*/
	{
		return;
	}
	int i=left;
	int j=right;
	int key=a[left];

	while(i<j)
	{
		while(i<j && key<=a[j])/*而寻找结束的条件就是,1,找到一个小于或者大于key的数(大于或小于取决于你想升
								序还是降序)2,没有符合条件1的,并且i与j的大小没有反转*/ 
			j--;	/*向前寻找*/
		a[i]=a[j];	/*找到一个这样的数后就把它赋给前面的被拿走的i的值(如果第一次循环且key是
					 a[left],那么就是给key)*/
		while(i<j && key>=a[i])		/*这是i在当组内向前寻找,同上,不过注意与key的大小关系停止循环和上面相反,
									因为排序思想是把数往两边扔,所以左右两边的数大小与key的关系相反*/
			i++;
		a[j]=a[i];		/*当在当组内找完一遍以后就把中间数key回归*/
	}
	a[i]=key;
	quicksort(a,left,i-1);/*最后用同样的方式对分出来的左边的小组进行同上的做法*/
	quicksort(a,i+1,right);/*用同样的方式对分出来的右边的小组进行同上的做法*/
							/*当然最后可能会出现很多分左右,直到每一组的i = j 为止*/
}

4、选择排序
思路是设有10个元素a[1]-a[10],将a[1]与a[2]-a[10]比较,若a[1]比a[2]-a[10]都小,则不进行交换。若a[2]-a[10]中有一个以上比a[1]小,则将其中最小的一个与a[1]交换,此时a[1]就存放了10个数中最小的一个。同理,第二轮拿a[2]与a[3]-a[10]比较,a[2]存放a[2]-a[10]中最小的数,以此类推。

#include<stdio.h>
int main()
{
	int i,j,t,a[10];
	printf("请输入10个整数:\n");
	for(i=0;i<10;i++)
		scanf("%d",&a[i]);    //输入10个整数存到数组里
	for(i=0;i<9;i++)
		for(j=i+1;j<10;j++)
			if(a[i]>a[j])     //如果前一个数比后一个大,则调换值
			{
				t=a[i];
				a[i]=a[j];
				a[j]=t;
			}
	for(i=0;i<10;i++)
		printf("%d  ",a[i]);
	printf("\n");

	return 0;
}

5、插入排序
插入排序是在一个已经有序的小序列的基础上,一次插入一个元素*
一般来说,插入排序都采用in-place在数组上实现。

具体算法描述如下:
⒈ 从第一个元素开始,该元素可以认为已经被排序
⒉ 取出下一个元素,在已经排序的元素序列中从后向前扫描
⒊ 如果该元素(已排序)大于新元素,将该元素移到下一位置
⒋ 重复步骤3,直到找到已排序的元素小于或者等于新元素的位置
⒌ 将新元素插入到下一位置中
⒍ 重复步骤2~5

#include<stdio.h>
int main()
{
	void insert(int a[],int first,int last);
	int i,a[10];	
	printf("输入10个整数:\n");
	for(i=0;i<10;i++)
		scanf("%d",&a[i]);
	insert(a,0,9);
	for(i=0;i<10;i++)
		printf("%5d",a[i]);	
	printf("\n");
	return 0;
}
void insert(int a[],int first,int last)
{
	int i,j,t;
	for(i=first+1;i<=last;i++)
	{
		t=a[i];
		j=i-1;
		while((j>=0) && (a[j]>t))//与已排序的数逐一比较,大于t时,该数移后
		{
			a[j+1]=a[j];
			j--;
		}
	//	if(j!=i-1)
            a[j+1]=t;
	}
}
Logo

旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐