【C++】十大经典排序算法详解

文章目录

排序算法总览

比较类排序算法

交换排序

一. 冒泡排序

二. 快速排序

三. 插入排序

四. 希尔排序

五. 选择排序

六. 堆排序

七. 归并排序

非比较类排序算法

一. 计数排序

二. 桶排序

三. 基数排序


本文译自 《【Python】十大经典排序算法详解》,由Python译为C++。

排序算法总览

十大排序算法可以分为比较类排序以及非比较类排序。

  • 比较类排序:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破O ( n l o g n ) O(nlogn)O(nlogn),因此也称为非线性时间比较类排序。
  • 非比较类排序:不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此也称为线性时间非比较类排序。
    在这里插入图片描述
    排序算法可以分为内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存。常见的内部排序算法有:插入排序、希尔排序、选择排序、冒泡排序、归并排序、快速排序、堆排序、基数排序等。用一张表格概括:
排序算法平均时间复杂度最好情况最坏情况空间复杂度排序方式稳定性
冒泡排序O ( n 2 ) O(n^2)O(n2)O ( n ) O(n)O(n)O ( n 2 ) O(n^2)O(n2)O ( 1 ) O(1)O(1)In-place稳定
选择排序O ( n 2 ) O(n^2)O(n2)O ( n 2 ) O(n^2)O(n2)O ( n 2 ) O(n^2)O(n2)O ( 1 ) O(1)O(1)In-place不稳定
插入排序O ( n 2 ) O(n^2)O(n2)O ( n ) O(n)O(n)O ( n 2 ) O(n^2)O(n2)O ( 1 ) O(1)O(1)In- place稳定
希尔排序O ( n l o g n ) O(n log n)O(nlogn)O ( n l o g 2 n ) O(n log^2 n)O(nlog2n)O ( n l o g 2 n ) O(n log^2n)O(nlog2n)O ( 1 ) O(1)O(1)In- place不稳定
归并排序O ( n l o g n ) O(n log n)O(nlogn)O ( n l o g n ) O(nlogn)O(nlogn)O ( n l o g n ) O(nlogn)O(nlogn)O ( n ) O(n)O(n)Out-place稳定
快速排序O ( n l o g n ) O(n log n)O(nlogn)O ( n l o g n ) O(n log n)O(nlogn)O ( n 2 ) O(n^2)O(n2)O ( l o g n ) O(log n)O(logn)In-place不稳定
堆排序O ( n l o g n ) O(n log n)O(nlogn)O ( n l o g n ) O(nlogn)O(nlogn)O ( n l o g n ) O(nlogn)O(nlogn)O ( 1 ) O(1)O(1)In-place不稳定
计数排序O ( n + k ) O(n + k)O(n+k)O ( n + k ) O(n + k)O(n+k)O ( n + k ) O(n + k)O(n+k)O ( k ) O(k)O(k)Out-place稳定
桶排序O ( n + k ) O(n + k)O(n+k)O ( n + k ) O(n + k)O(n+k)O ( n 2 ) O(n^2)O(n2)O ( n + k ) O(n + k)O(n+k)Out- -place稳定
基数排序O ( n k ) O(nk)O(nk)O ( n k ) O(nk)O(nk)O ( n k ) O(nk)O(nk)O ( n + k ) O(n + k)O(n+k)Out-place稳定

比较类排序算法

交换排序

一. 冒泡排序

算法思想

冒泡排序(Bubble Sort)也是一种简单直观的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢"浮"到数列的顶端。

作为最简单的排序算法之一,冒泡排序给我的感觉就像 Abandon 在单词书里出现的感觉一样,每次都在第一页第一位,所以最熟悉。冒泡排序还有一种优化算法,就是立一个 flag,当在一趟序列遍历中元素没有发生交换,则证明该序列已经有序。但这种改进对于提升性能来说并没有什么太大作用。

算法步骤

1.比较相邻的元素。如果第一个元素比第二个元素大,就交换他们两个。
2.对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对。这步做完后,最后的元素会是最大的数。
3.针对所有的元素重复以上的步骤,除了最后一个。
4.持续每次对越来越少的元素重复上面的步骤,直到没有任何一对数字需要比较。

注:原始的冒泡是从后往前的

动图演示

代码实现

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

template <typename T>
//整数或浮点数皆可使用
void bubble_sort(T a[],int len) {
    for(int i=0; i<len-1; i++)
        for(int j=0; j<len-1-i; j++)
        	if(a[j]>a[j+1]) swap(a[j],a[j+1]);
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	bubble_sort(a,n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

二. 快速排序

算法思想

快速排序是由东尼·霍尔所发展的一种排序算法。在平均状况下,排序 n 个项目要 O ( n l o g n ) Ο(nlogn)O(nlogn) 次比较。在最坏状况下则需要 O ( n 2 ) Ο(n^2)O(n2) 次比较,但这种状况并不常见。事实上,快速排序通常明显比其他 O ( n l o g n ) Ο(nlogn)O(nlogn) 算法更快,因为它的内部循环(inner loop)可以在大部分的架构上很有效率地被实现出来。快速排序使用分治法(Divide and conquer)策略来把一个串行(list)分为两个子串行(sub-lists)。快速排序又是一种分而治之思想在排序算法上的典型应用。本质上来看,快速排序应该算是在冒泡排序基础上的递归分治法。

算法步骤

1.从数列中挑出一个元素,称为 “基准”(pivot);
2.重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个分区退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作;
3.递归地(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序;

动图演示

代码实现

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	sort(a,a+n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}
  •  
#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

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

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	qsort(a,0,n-1);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

void qsort(int a[],int l,int r){
    if(l>=r) return;
    int i=l,j=r+1;
    int x=a[l];
    while(1) {
        //从左向右找比x大的值
        while(a[++i]<x)
            if(i==r) break;
        //从右向左找比x小的值
        while(a[--j]>x)
            if(j==l) break;
        if(i>=j) break;
        //交换i,j对应的值
        swap(a[i],a[j]);
    }
    //中枢值与j对应值交换
    swap(a[l],a[j]);
    qsort(a,l,j-1);
    qsort(a,j+1,r);
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	qsort(a,0,n-1);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

三. 插入排序

算法思想

插入排序是一种最简单直观的排序算法,它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。就像我们斗地主时,抽牌阶段会把抽到的牌插入到相应的位置中去,使手上的牌有序。
插入排序有个小优化叫做折半插入,就是往前寻找插入位置时,因为前面的数组全部有序,因此我们用二分查找法来寻找插入位置。

算法步骤

将第一待排序序列第一个元素看做一个有序序列,把第二个元素到最后一个元素当成是未排序序列。从头到尾依次扫描未排序序列,将扫描到的每个元素插入有序序列的适当位置。(如果待插入的元素与有序序列中的某个元素相等,则将待插入元素插入到相等元素的后面,保持相应顺序不变,插入排序是一个稳定的排序算法。)

动图演示

代码实现

def insertionSort(arr):
    for i in range(1,len(arr)):
        pos, insert_num = 0, arr[i]
        for j in range(i-1,-1,-1):
            if insert_num < arr[j]:
                arr[j+1] = arr[j]
            if insert_num >= arr[j]:
                arr[j+1] = insert_num
                pos = j+1
                break
        if pos == 0:
            arr[0] = insert_num

四. 希尔排序

算法思想

希尔排序,也称递减增量排序算法,是插入排序的一种更高效的改进版本。但希尔排序是非稳定排序算法。

希尔排序是基于插入排序的以下两点性质而提出改进方法的:

  • 插入排序在对几乎已经排好序的数据操作时,效率高,即可以达到线性排序的效率
  • 但插入排序一般来说是低效的,因为插入排序每次只能将数据移动一位

希尔排序的基本思想是:先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,待整个序列中的记录"基本有序"时,再对全体记录进行依次直接插入排序。

算法步骤

1.选择一个增量序列 t 1 , t 2 , … … , t k t_1,t_2,……,t_kt1​,t2​,……,tk​,其中 t i > t j t_i > t_jti​>tj​, t k = 1 t_k = 1tk​=1;
2.按增量序列个数 k kk,对序列进行 k kk 趟排序;
3.每趟排序,根据对应的增量 t i t_iti​,将待排序列分割成若干长度为 m mm 的子序列,分别对各子表进行直接插入排序。仅增量因子为 1 11 时,整个序列作为一个表来处理,表长度即为整个序列的长度。

动图演示

代码实现

只需要将插入排序稍微修改一下,就可以得到希尔排序。

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

void insertion_sort(int a[],int len){
	for(int i=0; i<len; i++) {
		//将a[i]插入到a[i-1],a[i-2],a[i-3]……之中
		for(int j=i-1; j>=0 && a[j+1]<a[j]; j--)
			swap(a[j],a[j+1]);
	}
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	insertion_sort(a,n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

五. 选择排序

算法思想

选择排序是一种简单直观的排序算法,无论什么数据进去都是 O(n²) 的时间复杂度。所以用到它的时候,数据规模越小越好。唯一的好处可能就是不占用额外的内存空间了吧。

算法步骤

首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置。
再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。
重复第二步,直到所有元素均排序完毕。

动图演示

代码实现

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

void select_sort(int a[],int len){
	for(int i=0; i<len; i++) 
		for(int j=i+1; j<len; j++) 
			if(a[j]<a[i]) swap(a[j],a[i]);
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	select_sort(a,n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

  •  
#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

void select_sort(int a[],int len){
	for(int i=0; i<len; i++) {
		int x=i;
		for(int j=i+1; j<len; j++) 
			if(a[x]>a[j]) x=j;
		if(i!=x) swap(a[x],a[i]);
	}
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	select_sort(a,n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

六. 堆排序

算法思想

堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。堆排序可以说是一种利用堆的概念来排序的选择排序。分为两种方法:

  • 大顶堆:每个节点的值都大于或等于其子节点的值,在堆排序算法中用于升序排列
  • 小顶堆:每个节点的值都小于或等于其子节点的值,在堆排序算法中用于降序排列

堆排序的平均时间复杂度为 Ο(nlogn),利用堆的特性,其实我们可以很方便的得到一个未排序数组中的Top K元素。

算法步骤

1.创建一个堆 H[0……n-1];
2.把堆首(最大值)和堆尾互换;
3.把堆的尺寸缩小 1,并调用 shift_down(0),目的是把新的数组顶端数据调整到相应位置
4.重复步骤 2,直到堆的尺寸为 1。

动图演示

代码实现

#pragma GCC optimize(3,"Ofast","inline")
#pragma G++ optimize(3,"Ofast","inline")

#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>

#define RI                 register int
#define re(i,a,b)          for(RI i=a; i<=b; i++)
#define ms(i,a)            memset(a,i,sizeof(a))
#define MAX(a,b)           (((a)>(b)) ? (a):(b))
#define MIN(a,b)           (((a)<(b)) ? (a):(b))

using namespace std;

typedef long long LL;

const int N=1e6+5;

int n;
int a[N];

void max_heapify(int a[],int start,int end) {
	//建立父节点指标和子节点指标
	int dad=start;
	int son=dad<<1+1;
	while(son<=end) { 
		//若子节点指标在范围内才做比较
		if(son+1<=end && a[son]<a[son+1]) son++;
		//先比较两个子节点大小,选择最大的
		if(a[dad]>a[son]) return;//如果父节点大於子节点代表调整完毕,直接跳出函数
			else { //否则交换父子内容再继续子节点和孙节点比较
				swap(a[dad],a[son]);
				dad=son;
				son=dad<<1+1;
			}
	}
}
 
void heap_sort(int a[],int len) {
	//初始化,i从最後一个父节点开始调整
	for(int i=len>>1-1; i>=0; i--)
		max_heapify(a,i,len-1);
	//先将第一个元素和已经排好的元素前一位做交换,再从新调整(刚调整的元素之前的元素),直到排序完毕
	for(int i=len-1; i>0; i--) {
		swap(a[0],a[i]);
		max_heapify(a,0,i-1);
	}
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	heap_sort(a,n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
    printf("\n");
	return 0;
}

七. 归并排序

算法思想

归并排序(Merge sort)是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。作为一种典型的分而治之思想的算法应用,归并排序的实现由两种方法:

自上而下的递归(所有递归的方法都可以用迭代重写,所以就有了第 2 种方法);
自下而上的迭代;

算法步骤

二路归并排序:
1.申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列;
2.设定两个指针,最初位置分别为两个已经排序序列的起始位置;
3.比较两个指针所指向的元素,选择相对小的元素放入到合并空间,并移动指针到下一位置;
4.重复步骤 3 直到某一指针达到序列尾;
5.将另一序列剩下的所有元素直接复制到合并序列尾。

动图演示

代码实现

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000],a1[1000];

void Merge(int a[],int a1[],int l,int mid,int r) {
    int i=l,j=mid+1,k=l;
    while(i<=mid && j<=r) {
        if(a[i]<=a[j]) a1[k++]=a[i++];
        	else a1[k++]=a[j++];
    }
    if(i<=mid) while(i<=mid) a1[k++]=a[i++];
    	else while(j<=r) a1[k++]=a[j++];
    for(int i=l; i<=r; i++)
        a[i]=a1[i];
}

void merge_sort(int a[],int a1[],int l,int r) {
    if(l<r) {
        int mid=(l+r)>>1;
        merge_sort(a,a1,l,mid);
        merge_sort(a,a1,mid+1, r);
        Merge(a,a1,l,mid,r);
    }
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	merge_sort(a,a1,0,n-1);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}

多路归并排序的思路参考上述代码,无非就是多了几个有序数组合并而已。

非比较类排序算法

一. 计数排序

算法思想

计数排序的核心在于将输入的数据值转化为键存储在额外开辟的数组空间中。作为一种线性时间复杂度的排序,计数排序要求输入的数据必须是有确定范围的整数。当输入的元素是 n 个 0 到 k 之间的整数时,它的运行时间是 O(n + k)。计数排序不是比较排序,排序的速度快于任何比较排序算法。
由于用来计数的数组C的长度取决于待排序数组中数据的范围(等于待排序数组的最大值与最小值的差加上1),这使得计数排序对于数据范围很大的数组,需要大量时间和内存。例如:计数排序是用来排序0到100之间的数字的最好的算法,但是它不适合按字母顺序排序人名。但是,计数排序可以用在基数排序中的算法来排序数据范围很大的数组。
通俗地理解,例如有 10 个年龄不同的人,统计出有 8 个人的年龄比 A 小,那 A 的年龄就排在第 9 位,用这个方法可以得到其他每个人的位置,也就排好了序。当然,年龄有重复时需要特殊处理(保证稳定性),这就是为什么最后要反向填充目标数组,以及将每个数字的统计减去 1 的原因。

算法步骤

1.找出待排序的数组中最大和最小的元素
2.统计数组中每个值为i的元素出现的次数,存入数组C的第 i 项
3.对所有的计数累加(从C中的第一个元素开始,每一项和前一项相加)
4.反向填充目标数组:将每个元素 i 放在新数组的第C(i)项,每放一个元素就将C(i)减去1

动图演示

代码实现

#include <bits/stdc++.h>
using namespace std;

int n,k=1e5;
int a[100000],c[100000],ans[100000];

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) {
		scanf("%d",&a[i]);
		c[a[i]]++;
	}
	for(int i=1; i<=k; i++) c[i]+=c[i-1];
	for(int i=n-1; i>=0; i--) ans[--c[a[i]]]=a[i];
	for(int i=0; i<n; i++) printf("%d ",ans[i]);
	printf("\n");
    return 0;
}

二. 桶排序

算法思想

桶排序是计数排序的升级版。它利用了函数的映射关系,高效与否的关键就在于这个映射函数的确定。为了使桶排序更加高效,我们需要做到这两点:

在额外空间充足的情况下,尽量增大桶的数量
使用的映射函数能够将输入的 N 个数据均匀的分配到 K 个桶中
同时,对于桶中元素的排序,选择何种比较排序算法对于性能的影响至关重要。

1.什么时候最快
当输入的数据可以均匀的分配到每一个桶中。

2.什么时候最慢
当输入的数据被分配到了同一个桶中。

示意图

元素分布在桶中:
在这里插入图片描述
然后,元素在每个桶中排序:
在这里插入图片描述

代码实现

#include <bits/stdc++.h>
using namespace std;

int n,k=1e5;
int c[100000];

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) {
		int x;
		scanf("%d",&x);
		c[x]++;
	}
	for(int i=0; i<=k; i++) {
		while(c[i]>0) printf("%d ",i),c[i]--;
	}
	printf("\n");
    return 0;
}

三. 基数排序

算法思想

基数排序是一种非比较型整数排序算法,其原理是将整数按位数切割成不同的数字,然后按每个位数分别比较。由于整数也可以表达字符串(比如名字或日期)和特定格式的浮点数,所以基数排序也不是只能使用于整数。

基数排序 vs 计数排序 vs 桶排序

基数排序有两种方法:

这三种排序算法都利用了桶的概念,但对桶的使用方法上有明显差异:

基数排序:根据键值的每位数字来分配桶;
计数排序:每个桶只存储单一键值;
桶排序:每个桶存储一定范围的数值;

动图演示

代码实现

#include <bits/stdc++.h>
using namespace std;

int n;
int a[1000];

void radix_sort(int a[],int size) {   
    int n;  
    for(int i=1; i<=100; i=i*10) {
        int tmp[1000][10];
        memset(tmp,0,sizeof(tmp));
        //建立一个1000行,10列的数组,每一列分别代表0~9位数,1000行代表能存放的总个数
        for(int j=0;j<size;j++) {
            n=(a[j]/i)%10;
            tmp[j][n]=a[j];
        }
        int k=0;
        for(int p=0; p<10; p++)
            for(int q=0; q<size; q++) {
                if(tmp[q][p]!=0)
                    a[k++]=tmp[q][p];
            }
    }
}

int main() {
	scanf("%d",&n);
	for(int i=0; i<n; i++) scanf("%d",&a[i]);
	radix_sort(a,n);
	for(int i=0; i<n; i++) printf("%d ",a[i]);
	printf("\n");
    return 0;
}
  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
C语言的经典算法。 第一篇 基础与提高  实例1利用库函数编写基本显示程序  实例2变量属性  实例3运算符与类型  实例4关于程序结构  实例5显示函数曲线图  实例6二分法选代的应用  实例7多变的立方体  实例8一维整型数组应用(1)  实例9一维整型数组应用(2)  实例10一维整型数组应用(3)  实例11一维整型数组应用(4)  实例12二维数组应用(1)——显示杨辉三角  实例13二维数组应用(2)——魔方阵  实例14字符数组应用(1)——逻辑判断  实例15字符数组应用(2)——数据模拟  实例16二维数组应用——字符比较   实例17利用指针进行数据处理  实例18指针与字符串  实例19利用指针处理二维数组  实例20一级指针  实例21利用指针传递参数值  实例22结构体的应用  实例23链表的应用(1)  实例24链表的应用(2)  实例25链表的应用(3)  实例26共用体的应用  实例27枚举类型应用  实例28位运算  买例29义件加密  实例30文件的按记录随机读写 第二篇 图形与多媒体  实例31改变文字背景色  实例32及本颜色设置  实例33制作表格  实例34制作多样的椭圆  实例35美丽的透视图形  实例36错位窗口  实例37能移动的矩形  实例38多变的填充矩形  实例39黄黑相间的矩形与圆  实例40六叶图案  实例41特殊图案  实例42国际象棋棋盘  实例43制作楼梯  实例44使用线类型函数设置多个汉字  实例45彩色群点  实例46饼图  买例47产品折线图  实例48直方图  实例49变大变色的拒形与国  实例50多变的填充多边形  实例51流星球  实例52小球动态碰撞  买倒53多,曲线  实例54多变的圆与环  实例55优美的球体  实例56运动的小车  实例57统计动画消失次数  实例58运行的时钟  实例59直升飞机  实例60演绎“生命游戏”  实例61猜猜看  买例62艺术清屏  买倒63制作火焰  实例64动态绘制256条不同颜色的直线  实例65红绿蓝三原色渐变 第三篇 综合小程序  实例66两个矩阵相乘  实例67艺术钟  实例68家庭财务管理小程序  实例69用系统时间实现随机数  实例70闪动的多彩圆  实例71检查系统有无鼠标  实例72圆形光盘与矩形  实例73动态渐变图案  实例74往返两地间的小车  实例75飘扬的红旗  实例76显示蓝天白云图形  实例77百叶窗 第四篇 计算机等级考试上机试题(二级) 第五篇 计算机等级考试上机试题(三级)
C语言中的不稳定的排序算法有冒泡排序、选择排序和快速排序。其中,冒泡排序是一种简单直观的排序算法,它的原理是从第一个元素开始,与后面的元素逐个比较,如果顺序不对就交换,直到没有可比较的元素为止。选择排序也是一种简单直观的排序算法,它的工作原理是每次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,然后再从剩余的未排序元素中寻找到最小(大)元素,放到已排序的序列的末尾。以此类推,直到全部待排序的数据元素的个数为零。而快速排序是一种高效的排序算法,它的基本思想是通过一趟排序将待排序的记录分割成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,然后再分别对这两部分记录进行排序,直到整个序列有序。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [c++十大经典排序算法](https://blog.csdn.net/zhoujiajie0521/article/details/122183332)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [C++八种排序算法万字详解](https://blog.csdn.net/m0_64036070/article/details/125106357)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hbqjzx

你的鼓励将是我分享的最大动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值