赞
踩
目录
排序:排序是指一串记录,按照其中的某个或某些关键字的大小,使他们递增或递减的进行排列起来的操作。(将无序变成有序)
内部排序:数据元素全部放在内存中的排序。
外部排序:数据元素太多不能同时放在内存中,根据排序过程的要求不能在内外存之间移动数据的排序。
原理:最简单的排序,将元素逐个插入到已经排好序的有序列表中,从而得到一个新的、容量增1的有序表。
图示讲解:
实现代码:
- #define _CRT_SECURE_NO_WARNINGS 1
- #include<stdio.h>
-
- void Print(int arr[], int n)//结果打印函数
- {
- int i = 0;
- for (i = 0; i < n; i++)
- {
- printf("%d ", arr[i]);
- }
- printf("\n");
- }
- void Sort_arr(int arr[], int n)
- {
- int tmp = 0;
- int i = 0;
- int j = 0;
- for (i = 1; i < n; i++)
- {
- tmp = arr[i];
- for (j = i - 1; j >= 0; j--)
- {
- if (tmp > arr[j])
- {
- break;
- }
- else
- {
- arr[j + 1] = arr[j];
- }
- }
- arr[j + 1] = tmp;
- Print(arr, 10); //打印排序过程
- }
-
- }
- int main()
- {
- int a[10] = { 12,18,2,6,8,9,34,51,-2,7 };
- Sort_arr(a, 10);
- Print(a, 10);
-
- return 0;
- }
运行结果:
优点:直接插入排序在数组量较小,数组较为整齐时较快
缺点:不稳定,要是出现较小的数字在数组的后边,则会使排序变复杂
原理:希尔排序是对插入排序的优化,基本思路是先选定一个整数作为增量,把待排序文件中的所有数据分组,以每个距离的等差数列为一组,对每一组进行排序,然后将增量缩小,继续分组排序,重复上述排序,直到增量为1时,排序完正好有序。整体原理是每一对分组进行排序后,整个数据就会更接近有序,当增量缩小为1 时就是插入排序,但是现在的数组非常的接近有序,移动的数据少,因此效率高,因此希尔排序又叫缩小增量排序。
图示讲解:
根据数组长度7,进行求找增量(gap)值
计算出第二轮增量为1,进行直接排序
(会进行多次预排序,只要gap>1就是预排序,当gap==1时就是直接插入排序)
排出:1、2、3、5、6、8、8
实现代码:
- //希尔排序
- #define _CRT_SECURE_NO_WARNINGS 1
- #include<stdio.h>
-
- void ShellSort(int* arr, int sz)
- {
- int count = 0;//为了看希尔排序和直接插入排序的性能比较
- int gap = sz ;//设置排序的间隔
- while (gap > 1)
- {
- //一定要保证gap最后进来循环后为1,所以对此加1
- gap = gap / 3 + 1;//gap大于1为预排序,gap==1,为直接插入排序
- for (int i = 0; i < sz - gap; i++)//这里不是一次性把一组排完,而是挨个往后,一组一个轮流排
- {
- int end = i;
- int tmp = arr[end + gap];
- while (end >= 0)
- {
- if (tmp < arr[end])
- {
- arr[end + gap] = arr[end];
- end -= gap;
- count++;
- }
- else {
- break;
- }
- }arr[end + gap] = tmp;
- }
-
-
- }
- printf("希尔插入排序后移次数count=%d", count);
- }
- void Print(int* arr,int sz)
- {
- for (int i = 0; i < sz; i++)
- {
- printf("%d ", arr[i]);
- }
- }
-
- void test()
- {
- int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
- int sz = sizeof(arr) / sizeof(arr[1]);
- ShellSort(arr, sz);
- printf("\n");
- Print(arr, sz);
- }
- int main()
- {
- test();
- return 0;
- }
运行结果:
优点:在对几乎已经排好序的数据进行操作中效率较高,可以达到线性排序的效率
缺点:希尔排序一般情况下是低效的,因为插入排序每次只能将数据移动一位
原理: 在一组需要排序的数组中,对俩俩数据之间顺序与对应顺序相反时,交换数据。基本思想是从i=0位置到 i=n-1每次通过内循环找出i位置到n-1位置的最大(小)值
图示讲解:
实现代码:
- #define _CRT_SECURE_NO_WARNINGS 1
- #include <stdio.h>
-
- void Print(int* arr, int n)
- {
- int i = 0;
- for (i = 0; i < n - 1; i++) //打印结果
- {
- printf("%d ", arr[i]);
- }
- printf("\n");
- }
-
- void Sort_arr(int* arr, int n)
- {
- int i = 0;
- int j = 0;
- int min = 0;
- for (i = 0; i < n - 1; i++)
- {
- min = i;
- for (j = i + 1; j < n; j++)
- {
- if (arr[j] < arr[min]) //这里是找最小的那个数,然后记下该下标
- {
- min = j;
- }
- }
-
- if (i != min) //这里如果i==min时,就不需要交换
- {
- int tmp = arr[i];
- arr[i] = arr[min];
- arr[min] = tmp;
- }
- Print(arr, 10); //打印排序过程
- }
- }
-
- int main()
- {
-
- int arr[10] = { 2,6,1,-2,-4,9,19,37,-15,0 };
- Sort_arr(arr, 10); //实现排序的函数
- printf("排序后的数组:\n");
-
- Print(arr, 10); //实现打印的函数
-
- return 0;
- }
运行结果:
原理:堆排序其实是一种树状排序,以二叉堆为例是一颗完全二叉树,(除最后一层外每一层都有俩个子节点,且非满的二叉数子节点也在左边)二叉树满足每个节点都大于等于他的子节点(大顶堆)或每个节点都小于等于他的子节点(小顶堆),根据堆的定义可以得到堆满足顶点一定是整个序列的最大值(大顶堆)或者最小值(小顶堆)。
图解:
排升序建大堆,排降序建小堆
索引值下标:i
双亲结点的下标:parent =(i-1)/2
左孩子结点的下标 child1 = 2*i+2
右孩子结点的下标:child2 = 2*i+2
代码实现:
- #include <stdio.h>
- /*
- parent(i) = (i-1)/2
- left child (i) = 2*i + 1
- right child (i) = 2*i + 2
- */
-
- void swap(int *a, int *b)
- {
- int tmp;
- tmp = *a;
- *a = *b;
- *b = tmp;
- return;
- }
-
- void shiftUp(int arr[], int n, int k)
- {
- while ((k - 1) / 2 >= 0 && arr[k] > arr[(k - 1) / 2])
- {
- swap(&arr[k], &arr[(k - 1) / 2]);
- k = (k - 1) / 2;
- }
- return;
- }
-
- void shiftDown(int arr[], int n, int k)
- {
- int j = 0;
- while (2 * k + 1 < n)
- {
- j = 2 * k + 1;
- if (j + 1 < n && arr[j] < arr[j + 1])
- {
- j++;
- }
- if (arr[k] < arr[j])
- {
- swap(&arr[k], &arr[j]);
- k = j;
- }
- else
- {
- break;
- }
- }
- return;
- }
-
- void heapSort(int arr[], int n)
- {
- int i = 0;
- for (i = (n - 1 - 1) / 2; i >= 0; i--)
- {
- shiftDown(arr, n, i);
- }
- for (i = n - 1; i > 0; i--)
- {
- swap(&arr[0], &arr[i]);
- shiftDown(arr, i, 0);
- }
- return;
- }
-
- void printArray(int arr[], int n)
- {
- int i;
- for (i = 0; i < n; i++)
- {
- printf("%d ", arr[i]);
- }
- printf("\n");
- return;
- }
- void main()
- {
- int arr[10] = {1, 5, 9, 8, 7, 6, 3, 4, 0, 2};
-
- printArray(arr, 10);
- heapSort(arr, 10);
- printArray(arr, 10);
- }
运行结果:
原理:冒泡排序是最简单的排序算法,思路简单容易理解,是一种简单的交换排序,基本思想是俩俩比较相邻的关键字,如果反序则交换直到没有反序的记录为止。
图解:
代码实现:
- #define _CRT_SECURE_NO_WARNINGS 1
- #include <stdio.h>
-
- #define ARR_LEN 255 /*数组长度上限*/
-
- void bubble_Sort(int* arr, int len)
- {
- int i, flag, temp;
- do
- {
- flag = 0;
- for (i = 0; i < len - 1; i++)
- {
- if (arr[i] > arr[i + 1])
- {
- temp = arr[i];
- arr[i] = arr[i + 1];
- arr[i + 1] = temp;
- flag = i + 1;
- }
- }
- len = flag;
-
- } while (flag);
- }
-
- int main()
- {
-
- int len = 0;
- int arr[ARR_LEN] = { 0 };
-
- printf("please input arr len:");
- scanf("%d", &len);
-
- printf("please input arr member......\n");
- for (int j = 0; j < len; j++)
- {
- scanf("%d", &arr[j]);
- }
- puts("");
-
- printf("sort after is ......\n");
- bubble_Sort(arr, len);
- for (int j = 0; j < len; j++)
- {
- printf("%d ", arr[j]);
- }
- putchar('\n');
-
- return 0;
- }
运行结果:
原理:基本思想是通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。
图解:
代码实现:
- #include <stdio.h>
- #include <stdlib.h>
-
- #define N 10
-
- //快速排序法
- int quick_sort(int *a, int low, int high)
- {
- int i = low; //第一位
- int j = high; //最后一位
- int key = a[i]; //将第一个数作为基准值-- 先找到一个基准值
-
- //进行排序---> 最终结果就是 左面的 都比基准值小 ,右面的都比 基准值大,所以这是所有循环的结束条件
- while (i < j)
- {
- //下面的循环执行的条件是 如果右面的比基准值大,就赋一下值,否则继续向前移动
- //---如果直接把循环写成下面这样---
- //while(a[j] >= key) //如果下面的不写这个i<j,这个就出错、越界,并且排序不准--理由:
- //如果i<j,并且: 右面的值 大于 基准值 时,j往前移动一个
- //i 跟 j 的可能情况 只有 i<j i==j
- while(i < j && a[j] >= key)//i<j 是 当前while循环的结束条件,如果没有这个,i会大于j,出现越界,错误
- {
- j--;//继续走
- }//如果不成立,也就是 a[j] <= key;右面的比key小了,那就换个位置
- //把a[j]的数据给a[i]
- a[i] = a[j];
-
- //将事先保存好的基准值与左边的值进行比较,如果基准值大,保持不变,i往前
- //然后 判断一下这个新的a[i],也就是之前的a[j]跟key值的关系---> 一定是 a[i]<key
- //所以把i向前移动一下,i++
- while(i < j && a[i] <= key)
- {
- i++;
- }
- //移动完以后,把新的位置的a[i]的数值 给刚才的 a[j],然后开始下一次循环
- a[j] = a[i];
- }
-
- //跳出循环,将基准值放入数据a[i]中
- a[i] = key;
- //对基准值左边 的所有数据 再次进行快速查找(递归)
- if (i-1 > low)
- {
- quick_sort(a, low, i-1);
- }
-
- //对基准值右边的所有数据再次进行快速查找(递归)
- if (i+1 < high)
- {
- quick_sort(a, i+1, high);
- }
-
- return 0;
- }
-
- int main(int argc, const char *argv[])
- {
- int a[N] = {1, 3, 5, 7, 9, 2, 4, 6, 8, 0};//先整了个数组,初始化了一堆数
-
- int i = 0;
- printf("排序前:\n");
- for(i = 0; i < N; i++)
- {
- printf("%d ", a[i]);
- }
- putchar(10);
-
- //调用-快排
- quick_sort(a, 0, N-1);//数组,0 ,9
-
- printf("排序后:\n");
- for(i = 0; i < N; i++)
- {
- printf("%d ", a[i]);
- }
- putchar(10);
-
- return 0;
- }
运行结果:
原理:归并排序是基于归并操作的一种排序算法,归并操作的原理就是将一组有序的子序列合并成一个完整的有序序列,即首先需要把一个序列分成多个有序的子序列,通过分解到每个子序列只有一个元素时,每个子序列都是有序的,在通过归并各个子序列得到一个完整的序列。
图解:
代码:
- #include <stdio.h>
- //实现分割操作的函数
- void merge_sort(int* arr, int p, int q);
- //实现归并操作的函数
- void merge(int* arr, int p, int mid, int q);
-
- int main() {
- int i = 0;
- int arr[8] = { 7,5,2,4,1,6,3,0 };
- //对 arr 数组中第 1 至 8 个元素进行归并排序
- merge_sort(arr, 1, 8);
- while (i < 8)
- {
- printf("%d ", arr[i]);
- i++;
- }
- return 0;
- }
- //实现分割操作的函数,[p,q] 用于指定归并排序的区域范围,
- void merge_sort(int* arr, int p, int q) {
- int mid;
- if (arr == NULL || p > q || p == q) {
- return ;
- }
- mid = (p + q) / 2;
- //将 [p,q] 分为[p,mid] 和 [mid+1,q] 区域
- merge_sort(arr, p, mid);
- merge_sort(arr, mid + 1, q);
- //对分好的 [p,mid] 和 [mid,q] 进行归并操作
- merge(arr, p, mid, q);
- }
- //实现归并操作的函数,归并的 2 个区域分别为 [p,mid] 和 [mid+1,q]
- void merge(int* arr, int p, int mid, int q) {
- int i,j,k;
- int leftarr[100], rightarr[100];
- int numL = mid - p + 1;
- int numR = q - mid;
- //将 arr 数组中 [p,mid] 区域内的元素逐一拷贝到 leftarr 数组中
- for (i = 0; i < numL; i++) {
- leftarr[i] = arr[p - 1 + i];
- }
- //将 arr 数组中 [mid+1,q] 区域内的元素逐一拷贝到 rightarr 数组中
- leftarr[i] = 2147483647;
- for (i = 0; i < numR; i++) {
- rightarr[i] = arr[mid + i];
- }
- rightarr[i] = 2147483647;
- i = 0;
- j = 0;
- //逐一比较 leftarr 和 rightarr 中的元素,每次将较小的元素拷贝到 arr 数组中的 [p,q] 区域内
- for (k = p; k <= q; k++) {
- if (leftarr[i] <= rightarr[j]) {
- arr[k - 1] = leftarr[i];
- i++;
- }
- else {
- arr[k - 1] = rightarr[j];
- j++;
- }
- }
- }
运行结果:
原理:计数排序又称为鸽巢原理,是对哈希直接定制法变形应用,是一种稳定的算法,它的优势在于在对一定范围内的整数排序时,它的复杂度为Ο(n+k)(其中k是整数的范围),快于任何比较排序算法,当然这是一种牺牲空间换取时间的做法,而且当O(k)>O(n*log(n))的时候其效率反而不如基于比较的排序(基于比较的排序的时间复杂度在理论上的下限是O(n*log(n)),如归并排序,堆排序)
图解:
代码:
- #include<stdio.h>
- #include<assert.h>
- #include<stdlib.h>
- //计数排序
- void CountSort(int* a, int len)
- {
- assert(a);
- //通过max和min计算出临时数组所需要开辟的空间大小
- int max = a[0], min = a[0];
- for (int i = 0; i < len; i++) {
- if (a[i] > max)
- max = a[i];
- if (a[i] < min)
- min = a[i];
- }
- //使用calloc将数组都初始化为0
- int range = max - min + 1;
- int* b = (int*)calloc(range, sizeof(int));
- //使用临时数组记录原始数组中每个数的个数
- for (int i = 0; i < len; i++) {
- //注意:这里在存储上要在原始数组数值上减去min才不会出现越界问题
- b[a[i] - min] += 1;
- }
- int j = 0;
- //根据统计结果,重新对元素进行回收
- for (int i = 0; i < range; i++) {
- while (b[i]--) {
- //注意:要将i的值加上min才能还原到原始数据
- a[j++] = i + min;
- }
- }
- //释放临时数组
- free(b);
- b = NULL;
- }
- //打印数组
- void PrintArray(int* a, int len)
- {
- for (int i = 0; i < len; i++) {
- printf("%d ", a[i]);
- }
- printf("\n");
- }
- int main()
- {
- int a[] = { 3, 4, 3, 2, 1, 2, 6, 5, 4, 7 };
- printf("排序前:");
- PrintArray(a, sizeof(a) / sizeof(int));
- CountSort(a, sizeof(a) / sizeof(int));
- printf("排序后:");
- PrintArray(a, sizeof(a) / sizeof(int));
- system("pause");
- return 0;
- }
运行结果:
在实际应用中,选择合适的排序算法可以根据数据规模和特点来进行判断。以下是一些常见的选择排序算法的方法:
对于小规模的数据集,可以选择简单的排序算法,如插入排序、选择排序或冒泡排序。这些算法的时间复杂度较低,适用于小规模数据的排序。
对于大规模的数据集,可以选择更高效的排序算法,如归并排序、快速排序或堆排序。这些算法的时间复杂度较低,适用于大规模数据的排序。
如果数据集中存在大量重复元素,可以考虑使用计数排序或桶排序。这些算法可以通过统计元素出现的次数或将元素分配到不同的桶中来实现排序。
如果数据集中的元素是整数,并且范围较小,可以考虑使用基数排序。基数排序按照元素的位数进行排序,可以在一定程度上提高排序的效率。
在实际应用中,还可以根据具体的需求和场景选择合适的排序算法。例如,如果需要稳定的排序结果,可以选择归并排序或计数排序。
总之,选择合适的排序算法需要考虑数据规模、特点和排序需求等因素。根据不同的情况选择合适的排序算法可以提高排序的效率和准确性。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。