基本排序分类图:
关于排序的稳定性
在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次序保持不变,即在原序列中r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排序算法是稳定的;否则称为不稳定的。
一、 冒泡排序
冒泡排序的基本思想:每次比较两个相邻的元素,如果它们的顺序错误就把他们交换过来
冒泡排序的原理:每一趟只能确定将一个数归位,如果有n个数进行排序,只需将n-1个数归位,也就是说要进行n-1趟操作,而每一趟都需要从第1位开始进行相邻两个数的比较
1 #include <stdio.h>
2 #define MAX 7
3
4 int main(void)
5 {
6 int i, j, t;
7 int a[MAX] = {1, 5, 3, 7, 6, 4, 2};
8
9 //冒泡排序核心部分
10 for (i = 0; i < MAX - 1; i++) //n个数排序只需要n-1趟
11 {
12 for (j = 0;j < MAX - i; j++) //每一趟比较到n-i结束
13 {
14 if (a[j] < a[j + 1])//降序排列
15 {
16 t = a[j];
17 a[j] = a[j + 1];
18 a[j + 1] = t;
19
20 }
21 }
22 }
23 for (i = 0; i < MAX; i++)
24 {
25 printf("%d ", a[i]);
26 }
27
28 return 0;
29 }
冒泡排序的核心部分是双重嵌套循环,冒泡排序的时间复杂度是O(N2),这个一个非常高的时间复杂度
1 #include <stdio.h>
2 #define MAX 7
3
4 int main(void)
5 {
6 int i, j, t;
7 int flag = 0;
8 int n = 0;
9 int a[MAX] = {1, 5, 3, 7, 6, 4, 2};
10
11 //冒泡排序核心部分
12 for (i = 0; i < MAX - 1; i++) //n个数排序只需要n-1趟
13 {
14 flag = 0;
15 for (j = 0;j < MAX - i; j++) //每一趟比较到n-i结束
16 {
17 n++;
18 if (a[j] < a[j + 1])
19 {
20 t = a[j];
21 a[j] = a[j + 1];
22 a[j + 1] = t;
23 flag = 1;
24 }
25 }
26 if (flag == 0) /* 一次交换都没有,说明已经是有序序列*/
27 {
28 break; /* 跳出整个循环 */
29 }
30 }
31 for (i = 0; i < MAX; i++)
32 {
33 printf("%d ", a[i]);
34 }
35 printf("
比较次数%d
", n);
36
37 return 0;
38 }
优化前:
优化后:
二、快速排序
快速排序是基于二分的思想,对冒泡排序的一种改进
快速排序基本思想:
通过一趟排序将要排序的数据分割成独立的两部分:分割点左边都是比它小的数,右边都是比它大的数。然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。
快速排序原理:第一步:设置两个指针left和right分别指向数组的头部和尾部,并且以头部的元素(6)为基准数
第二步:right指针先往左移动,找到小于基准数的元素就停下,然后移动left指针(想一下为什么是right先移动,不能是left先移动)
第三步:left指针往左移动,找到大于基准数的元素就停下,然后交换right和left指针所值元素的值
重复第二、三步,直到两个指针left和right重合
第四步:两个指针重合后将基准数(6)与两个指针指向的元素值(3)交换
到这时,第一轮排序结束,此时以基准数(6)为分界点,(6)左边的数都小于等于6,(6)右边的数都大于等于6,现在我们已经将原来的序列以(6)为分界点拆成了两个序列
左边的序列是3 1 2 5 4,右边的序列是9 7 10 8,接下来分别处理这两个序列,因为处理方法与上图相同,下面就不上图了(摸个鱼)
先处理3 1 2 5 4,以(3)为基准数,处理后结果为2 1 3 5 4
再处理(3)左边的数2 1,以(2)为基准数,处理后为1 2
再处理(3)右边的数5 4,以(5)为基准数,处理后为4 5
现在的序列为1 2 3 4 5 6 9 7 10 8,现在对9 7 10 8进行处理
因为处理方式与上面相同这里不再赘述,处理结果为7 8 9 10
最终序列为1 2 3 4 5 6 7 8 9 10,到此排序完全结束(快速排序的每一轮处理其实就是将这一轮的基准数归位,直到所有的数都归位为止)
1 #include <stdio.h>
2 void quicksort(int left, int right, int *a);
3
4 #define N 7
5 int main(void)
6 {
7 int i;
8 int a[N] = {5, 1, 7, 2, 4, 3, 6};
9
10 quicksort(0, N-1, a);
11
12 for(i = 0; i < N; i++)
13 {
14 printf("%d ", a[i]);
15 }
16
17 return 0;
18 }
19 void quicksort(int left, int right, int *a)
20 {
21 int i, j, t, temp;
22 if (left > right) //基线情况
23 {
24 return;
25 }
26
27 temp = a[left]; //存入基准数
28 i = left;
29 j = right;
30 while (i != j)
31 {
32 //先从右往左找
33 while (a[j] >= temp && j>i)
34 {
35 j--;
36 }
37 //从左往右找
38 while (a[i] <= temp && j>i)
39 {
40 i++;
41 }
42
43 if (i < j)
44 {
45 t = a[i];
46 a[i] = a[j];
47 a[j] = t;
48 }
49 }
50 a[left] = a[i]; //基准数归位
51 a[i] = temp;
52
53 quicksort(left, i - 1, a);
54 quicksort(i + 1, right, a);
55
56 return;
57 }
快速排序在最坏的情况下,仍可能是相邻的两个数进行交换,因此快速排序最差时间复杂度和冒泡排序是一样的,都是O(N2),它的平均时间复杂度为O(NlogN)