跳到主要内容

22、数据结构和算法 - 实战:排序算法

定义

 

排序的稳定性

假设ki = kj,且在排序前的序列中ri领先于rj(i < j)
如果排序后仍然ri领先,那么称所用的排序方法是稳定的;反之,称为不稳定。

影响因素

时间性能,辅助空间,算法的复杂性。都影响排序的算法性能。

排序算法

 

冒泡排序

基本思想:两两相邻记录的关键字,如果反序则交换,直到没有反序的记录为止。

要点:

  • 两两相邻的元素
  • 如果有n个元素,比较n-1次,每一轮减少一次比较
  • 从下往上比较,像冒泡泡一样。

原始冒泡排序的算法,这并不符合两两相邻的比较

#include <stdio.h>
void BubbleSort(int k[], int n)
{
   
     
	int temp;
	for(int i = 0;i< n-1; i++)
	{
   
     
		for(int j = i+1; j< n-1 ;j++)
		{
   
     
			if(k[i]> k[j])
			{
   
     
				temp = k[j];
				k[j] = k[i];
				k[i] = temp;
			}
		}
	}
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	BubbleSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

正宗的冒泡排序算法

#include <stdio.h>
void BubbleSort(int k[], int n)
{
   
     
	int temp, count1 = 0, count2 = 0;
	for(int i=0;i< n-1; i++)
	{
   
     
		for(int j = n-1; j > i ;j--)
		{
   
     
			count1++;
			if(k[i]> k[j])
			{
   
     
				count2++;
				temp = k[j - 1];
				k[j- 1 ] = k[j];
				k[j] = temp;
			}
		}
	}
	printf("总共进行了%d次比较,进行了%d移动\n",count1,count2);
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	BubbleSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

选择排序

基本思想:通过n-i次关键字间的比较,从n-i+1个记录中选出关键字最小的记录,并和第i个记录交换。
(类似原始的冒泡排序,但是区别在于,这个判断完了再交换,而不是立刻更改数据)

#include <stdio.h>
void SelectSort(int k[], int n)
{
   
     
	int temp, min;
	for(int i = 0;i< n-1; i++)
	{
   
     
		for(int j = i+1; j< n-1 ;j++)
		{
   
     
			min = i;
			if(k[min]> k[j])
			{
   
     
				min = j;
			}
		}
		if(min != i)
		{
   
     
			temp = k[min];
			k[min] = k[i];
			k[i] = temp;
		}
	}
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	SelectSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

效率相对来说比冒泡高一点

直接插入排序算法

基本操作将一个记录插入到已经排好序的有序表中,从而得到一个新的,记录数增加1的有序表。

#include <stdio.h>
void InsertSort(int k[], int n)
{
   
     
	int temp, min;
	for(int i = 1;i< n; i++)
	{
   
     
		if(k[i] < k[i-1])
		{
   
     
			temp = k[i];
			for(int j = i-1; k[j] > temp ;j--)
			{
   
     
				k[j + 1] = k[j];
			}
		k[j + 1] = temp;
		}		
	}
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	InsertSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

希尔排序

第一次跨度为4,第二次跨度为2,第三次跨度为0.
 
实现代码

#include <stdio.h>
void XierSort(int k[], int n)
{
   
     
	int temp, min;
	int gap = n;
	do
	{
   
     
		gap = gap/3 + 1
		for(int i = gap;i< n; i++)
		{
   
     
			if(k[i] < k[i-gap])
			{
   
     
				temp = k[i];
				for(int j = i-gap; k[j] > temp ;j-=gap)
				{
   
     
					k[j + gap] = k[j];
				}
			k[j + gap] = temp;
			}		
		}
	}while(gap > 1);
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	XierSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

堆排序

之前讲过的排序算法,比较次数比较多,导致算法复杂度增加。
堆排序:对选择排序进行改进,复杂度O(Nlogn),利于之前排序的结果。

大顶堆:每个节点都大于等于其左右孩子的节点。
小顶堆:每个节点都小于等于其左右孩子的节点。
 
要点:根结点一定是堆中结点最大或者最小的,如果按照层序遍历的方式进行编号,则结点之间满足如下关系
 
(左为大顶堆,右为小顶堆)下标i与2i和2i+1是双亲和子女关系
则存储结果如下
 
 
堆排序算法的基本思想:利用堆进行排序的算法

  • 将待排序的序列构造成一个大顶堆(小顶堆)
  • 此时,整个序列的最大值就是堆顶的根结点,将它移走(就是将其与堆数组的末尾元素交换,此时末尾元素就是最大值)
  • 然后将剩余的n-1个序列重新构造成一个堆,这样就会得到n个元素中的此大值。
  • 如此反复执行,得到一个有序序列
#include <stdio.h>
void HeapAdjust(int k[],int s, int n)
{
   
     
	int i, temp;
	temp = k[s];
	for(i = 2*s;i <=n;i *=2)
	{
   
     
		if(i< n && k[i] < k[i+1])
		{
   
     	
			i++;
		}
		if(temp >=k[i])
		{
   
     	
			break;
		}
		k[s] = k[i];
		s = i;
	}
	k[s] = temp;
}
void swap(int k[],int i, int j)
{
   
     
	int temp;
	temp = k[i];
	l[i] = k[j];
	k[j] = temp;
}
void HeapSort(int k[], int n)
{
   
     
	int i ;
	for(i = n/2; i> 0;i--)
	{
   
     
		HeapAdjust(k ,i , n);
	}
	for(i =n;i >1; i++)
	{
   
     
		swap(k, 1, i);
		HeapAdjust(k ,1 , i-1);
	}
}

int main()
{
   
     
	int i ,a[10] = {
   
     -1,5,2,6,0,3,9,1,7,4};
	HeapSort(a,9);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

归并排序

归并排序:就是利用归并的思想实现的排序方法,原理是假设初始序列有n个记录,可以看成是n个有序的子序列,每个子序列的长度为1,然后两两归并,得到n/2(向上取整)个长度为2或1 的有序子序列,再两两归并,如此重复,直到得到一个长度为n的有序序列为止,这种排序方法称为2路归并排序。
 

递归实现

#include <stdio.h>
//实现归并,把结果存在list1里面
void merging(int *list1,int list1_size, int *list2, int list2_size)
{
   
     
	int temp[MAXSIZE];
	int i, j ,k, m;

	while(i < list1_size && j< list2_size)
	{
   
     
		if(list1[i]<list2[j])
		{
   
     
			temp[k++] = list1[i++];
		}
		else
		{
   
     
			temp[k++] = list2[j++];
		}
	}
	while(i < list1_size)
	{
   
     
		temp[k++] = list1[i++];
	}
	while(j < list2_size)
	{
   
     
		temp[k++] = list1[j++];
	}
	for( m=0; m< (list1_size + list2_size);m++)
	{
   
     
		list1[m] = temp[m];
	}
}
void MergeSort(int k[], int n)
{
   
     
	if(n >1)
	{
   
     
	int *list1 = k;
	int list1_size = n/2;
	int *list2 = k + n/2;
	int list1_size = n - list1_size;
	
	MergeSort(list1 , list1_size);
	MergeSort(list2 , list2_size);

	merging(list1,list1_size,list2,list2_size);
	}
	
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	MergeSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

迭代实现

#include <stdio.h>
#define MAXSIZE  10
void MergeSort(int k[], int n)
{
   
     
	int i ,left_min,left_max,right_min,right_max;
	int *temp = (int *)malloc (n * sizeof(int));
	
	for(i = 1;i<n; i *=2)
	{
   
     
		for( left_min = 0;left_min < n-i; letf_min = right_max)
		{
   
     
			right_min = left_max = left_min + i;
			right_max = left_max + i;
			
			if(right_max >n)
			{
   
     
				right_max = n;
			}
			next = 0
			while(left_min <left_max && right_min <right_max)
			{
   
     
				if(k[left_min] < k[right_min]
				{
   
     
					temp[next++] = k[left_min];
				}
				else
				{
   
     
					temp[next++] = k[right_min];
				}
			}
			while(left_min <left_max)
			{
   
     
				k[--right_min] = k[--left_min];
			}
			while(next >0)
			{
   
     
				k[--right_min] =temp[--next];
			}
		}
	}
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	MergeSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

快速排序算法

#include <stdio.h>
void swap(int k[],int low , int high)
{
   
     
	int temp;
	temp = k[low];
	k[low] = k[high];
	k[high] = temp;
}
int Partition(int k[],int low, int high)
{
   
     
	int point;
	point = k[low];
	while(low <high)
	{
   
     
		while(low <high && k[high] >= point)
		{
   
     
			high--;
		}
		swap(k ,low,high);
		while(low <high && k[low] <= point)
		{
   
     
			low++;
		}
		swap(k ,low,high);
	}
	return low;
}
void QSort(int k[], int low, int high)
{
   
     
	int point;
	if(low< high)
	{
   
     
		point = Partition(k,low,high); //计算基准点计算,一般选取第一个作为基准点
		QSort(k,low,point-1);
		QSort(k,point +1,high);
	}
}
void QuickSort(int k[], int n)
{
   
     
	QSort(k,0,n-1);
}

int main()
{
   
     
	int i ,a[10] = {
   
     5,2,6,0,3,9,1,7,4,8};
	QuickSort(a,10);
	
	printf("排序后: \n");
	for(i = 0;i < 10; i++)
		printf("%d\t",a[i]);
	
	printf("\n");
}

总结