返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C语言完整实现12种排序算法(小结)
  • 203
分享到

C语言完整实现12种排序算法(小结)

2024-04-02 19:04:59 203人浏览 薄情痞子
摘要

目录1.冒泡排序2.插入排序3.折半插入排序4.希尔排序5.选择排序6.鸡尾酒排序7.堆排序8.快速排序9.归并排序10.计数排序11.桶排序12.基数排序1.冒泡排序 思路:比较相

1.冒泡排序

思路:比较相邻的两个数字,如果前一个数字大,那么就交换两个数字,直到有序。
时间复杂度O(n^2),稳定性:这是一种稳定的算法
代码实现:

void bubble_sort(int arr[],size_t len){
    size_t i,j;
    for(i=0;i<len;i++){    
        bool hasSwap = false;        //优化,判断数组是否已经有序,如果有序可以提前退出循环
        for(j=1;j<len-i;j++){        //这里j<len-i是因为最后面的肯定都是最大的,不需要多进行比较
            if(arr[j-1]>arr[j]){    //如果前一个比后一个大
                swap(&arr[j-1],&arr[j]);    //交换两个数据
                hasSwap = true;
            }    
        }
        if(!hasSwap){
            break;    
        }
    }
}

2.插入排序

思路:把一个数字插入一个有序的序列中,使之仍然保持有序,如对于需要我们进行排序的数组,我们可以使它的前i个数字有序,然后再插入i+1个数字,插入到合适的位置使之仍然保持有序,直到所有的数字有序。
时间复杂度:O(n^2) 稳定性:稳定的算法
代码实现:

void insert_sort(int arr[],int len){
    int i,j;
    for(i=1;i<len;i++){
        int key = arr[i];            //记录当前需要插入的数据
        for(j= i-1;i>=0&&arr[j]>key;j--){    //找到插入的位置
            arr[j+1] = arr[j];                //把需要插入的元素后面的元素往后移
        }
        arr[j+1] = key;        //插入该元素
    }
}

3.折半插入排序

思路:本质上是插入排序,但是通过半分查找法找到插入的位置,让效率稍微快一点。
时间复杂度:O(n^2),稳定性:稳定的算法。
代码实现:

void half_insert_sort(int arr[],int len){
    int i,j;
    for(i=1;i<len;i++){
        int key = arr[i];
        int left = 0;
        int right = i-1;
        while(left<=right){    //半分查找找到插入的位置
            int mid = (left+right)/2;
            if(key<arr[mid]){
                right = mid-1;    
            }else{
                left = mid+1;    
            }    
        }
        for(j=i-1;j>=left;j--){        //把后面的元素往后移
            arr[j+1]=arr[j];    
        }
        arr[j+1] = key;    //插入元素
    }
}

4.希尔排序

思路:先取一个正整数d1<n,把所有序号相隔d1的数组元素放一组,组内进行直接插入排序;然后取d2<d1,重复上述分组和排序操作;直至di=1,即所有记录放进一个组中排序为止。
时间复杂度:O(n^1.3) ,算法效率上大大提高 。稳定性:不稳定的算法。
代码实现:

void shell_sort(int arr[],int len){    //本质上也是一种插入排序,避免了大量数据的移动,在每一组排序过后,每个数据已经到了大致的位置。
    int i,j;
    int step=0;
    for(step = len/2;step>=1;step=step/2){    //分组  分为step组,对每组的元素进行插入排序
        for(i=step;i<len;i++){
            int key = arr[i];
            for(j=i-step;j>=0&&arr[j]>key;j=j-step){
                arr[j+step] = arr[j];    
            }    
            arr[j+step] = key;
        }
    }
}

5.选择排序

思路:通过循环找到最大值所在的位置,然后把最大值和最后一个元素进行交换,通过循环直到所有的数据有序。
时间复杂度:O(n^2) 稳定性:不稳定的算法
代码实现:

void select_sort(int arr[],size_t len){
    size_t i,j;
    for(i=0;i<len-1;i++){
        int max = 0;        //最大值下标
        for(j=1;j<len-i;j++){
            if(arr[max]<arr[j]){    //找到最大值的下标
                max = j;    
            }    
        }
        if(max!=j-1){    
            swap(&arr[max],&arr[j-1]);    //把最后一个元素和最大值进行交换
        }
    }
}

6.鸡尾酒排序

思路:选择排序的一种改进,一次循环直接找到最大值和最小值的位置,把最大值和最后一个元素进行交换,最小值和最前一个元素进行交换,所以最外层的循环只需要执行len/2次即可
时间复杂度:O(n^2) 稳定性:不稳定的算法
代码实现:

void cocktail_sort(int arr[],size_t len){
    size_t i,j;
    for(i=0;i<len/2;i++){
        int max = i;    //最大值下标
        int min = i;    //最小值下标
        for(j=i+1;j<len-i;j++){
            if(arr[max]<arr[j]){    //找到最大值下标
                max = j;    
            }    
            if(arr[min]>arr[j]){    //找到最小值下标
                min = j;    
            }
        }
        if(max!=j-1){
            swap(&arr[max],&arr[j-1]);        //交换最大值和未进行排序的最后一个元素
        }
        if(min == j-1){    //如果最小值在未进行排序的最后一个位置,那么经过最大值的交换,已经交换到了最大值所在的位置
            min = max;        //把最小值的坐标进行改变
        }
        if(min!=i){
            swap(&arr[i],&arr[min]);    //交换最小值和未进行排序的最前的元素
        }
    }
}

7.堆排序

思路:把数据进行大堆化,然后依次交换堆顶(最大值)和最后一个元素,在使堆顶重新大堆化,最后循环过后数组便有序。
过程:
最大堆调整(Max Heapify):将堆的末端子节点作调整,使得子节点永远小于父节点
创建最大堆(Build Max Heap):将堆中的所有数据重新排序
堆排序(HeapSort):移除位在第一个数据的根节点,并做最大堆调整的递归运算
时间复杂度:O(nlgn) 稳定性:不稳定的算法
实现代码:

void re_heap(int arr[],size_t index,size_t len){
    size_t child = 2*index+1;    //左节点坐标
    int key = arr[index];    //当前节点值
    while(child<len){
        if(child+1<len&&arr[child]<arr[child+1]){    //如果右节点存在且右节点的值比左节点大,那就child记录较大字节点的坐标
            child++;    
        }    
        if(arr[child]>key){    //如果子节点的值比根节点的值大
            arr[index] = arr[child];    //改变根节点的值
        }else{
            break;    
        }
        index = child;
        child = 2*index+1;
    }
    arr[index] = key;        //插入记录好的值
}
void heap_sort(int arr[],size_t len){
    int i;
    for(i=len/2;i>=0;i--){
        re_heap(arr,i,len);        //对第i个根节点进行大堆化
    }
    for(i=len-1;i>0;i--){
        swap(&arr[0],&arr[i]);    //交换第一个和最后一个元素
        re_heap(arr,0,i);    //对第一个元素进行大堆化
    }
}

8.快速排序

思路:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。
过程:
(1)首先设定一个分界值,通过该分界值将数组分成左右两部分
(2)将大于或等于分界值的数据集中到数组右边,小于分界值的数据集中到数组的左边。此时,左边部分中各元素都小于或等于分界值,而右边部分中各元素都大于或等于分界值。
(3)然后,左边和右边的数据可以独立排序。对于左侧的数组数据,又可以取一个分界值,将该部分数据分成左右两部分,同样在左边放置较小值,右边放置较大值。右侧的数组数据也可以做类似处理。
(4)重复上述过程,可以看出,这是一个递归定义。通过递归将左侧部分排好序后,再递归排好右侧部分的顺序。当左、右两个部分各数据排序完成后,整个数组的排序也就完成了。
时间复杂度:O(nlog2n) 稳定性:不稳定的算法
代码实现:

void quick_sort(int arr[],size_t left,size_t right){
    if(left>=right){    //如果只有一个元素,那就是有序的,返回
        return;    
    }
    int i = left;
    int j = right;
    int key = arr[left];    //基准值
    while(i<j){    //找到基准值的位置,使得基准值右边的元素都比基准值大,左边的元素都比基准值小
        while(i<j&&arr[j]>=key){    //从右边找一个比基准值小的数,
            --j;
        }
        arr[i] = arr[j];//把这个值放到基准值的位置处
        while(i<j&&arr[i]<=key){    //从左边找一个比基准值大的数
            ++i;    
        }
        arr[j] = arr[i];        //把这个元素放到j的位置
    }
    arr[i] = key;
    if(i-left>1)    //元素个数至少两个才进行递归调用,这样可以少一次递归
        quick_sort(arr,left,i-1);    //对基准值左边的元素进行排序
    if(right-i>1)
        quick_sort(arr,i+1,right);    //对基准值右边的元素进行排序
}

9.归并排序

思路:对于两个有序的子序列,可以把它们合并在一起,变成一个新的完全有序的序列,因此归并排序和快排差不多,都是递归的进行。
时间复杂度:O(nlog2n) 稳定性:稳定的算法
代码实现:

void merge(int arr[],int left,int right){
    int i,j,k;
    int mid = (left+right)/2;
    int len = mid-left+1;
    int *temp = malloc(sizeof(arr[0])*len);
    for(i=0;i<len;i++){
        temp[i] = arr[i+left];    //把这个数组的所有元素都复制到临时数组中
    }
    i=0,j=mid+1,k=left;
    while(i<len&&j<=right){
        if(temp[i]<arr[j]){    //把临时数组的元素和 [mid+1,right]这部分的元素一个一个的进行比较,如果谁小,那么arr里就存放谁的元素
            arr[k++] = temp[i++];    
        }else{
            arr[k++] = arr[j++];    
        }
    }
    while(i<len){    //如果temp这个数组的元素还没有全部遍历完,那就把temp后面的元素都复制到arr里面去,
    //因为arr[mid+1,right] 这部分的元素本来就是arr后面部分的有序的元素,所以如果arr[mid+1,right]这部分没有遍历完也没关系的,
        arr[k++] = temp[i++];    
    }
    free(temp);
}
void merge_sort(int arr[],int left,int right){
    if(left>=right){    //如果只有一个元素说明这个序列有序,那就返回
        return;    
    }    
    int mid = (left+right)/2;    //对两个有序的数组进行排序,
    merge_sort(arr,left,mid);    //对[left,mid]这个区间的元素进行排序
    merge_sort(arr,mid+1,right);    //对[mid+1,right]这个区间内的元素进行排序
    merge(arr,left,right);  //这个序列的[left,mid]为有序的序列 [mid+1,right]也为有序的序列
}

10.计数排序

思路:这是一种基于比较的算法,我们用一个大数组来存放这些数据,这些数据在这个大数组中的表现形式是以这个大数组的下标存在的,比如57,60,42这三个数字进行排序,那么用一个大数组,这个大数组的arr[57] = 1,arr[60] = 1,arr[42] = 1,然后遍历这个大数组就行了。
时间复杂度:O(n+k),其中这个k为数据的范围,所以计数排序最适合数据比较集中的数组排序。
稳定性:稳定的算法
代码实现:

void count_sort(int arr[],size_t len){
    int max = arr[0];    //最大值
    int min = arr[0];    //最小值
    size_t i;
    for(i=0;i<len;i++){
        if(max<arr[i]){    //找到最大值
            max =arr[i];    
        }
        if(min > arr[i]){    //找到最小值
            min = arr[i];    
        }
    }
    int cnt = max-min+1;        //范围
    int *prr = malloc(cnt*sizeof(int));    //申请临时空间
    for(i=0;i<cnt;i++){    //这个临时数组全部置0
        prr[i] = 0;    
    }
    for(i=0;i<len;i++){    //对需要进行排序的序列进行遍历
        prr[arr[i]-min]++;        //让下标为(arr[i]-min)的临时大数组的值+1
    }
    size_t j=0;
    for(i=0;i<cnt;i++){    //遍历这个临时数组
        while(prr[i]){    //如果这个数组下标为i的值不等于0
            arr[j++] = i+min;    //那就让需要进行排序的数组的值为i+min;
            --prr[i];
        }    
    }
    free(prr);        //释放掉申请的动态内存
}

11.桶排序

思路:工作的原理是将数组分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。桶排序是鸽巢排序的一种归纳结果。
这是一种以消耗大量空间来换取高效率的排序方式,
时间复杂度:O(N+C),其中C=N*(logN-logM),M为桶的数量。所以对于桶排序,桶的数量越多,其排序效率越高。
稳定性:稳定的算法
代码实现:
首先定义桶这个类型:

typedef struct Bucket{
    int vect[100];    //其实这里使用链表更好,但是我比较懒,就懒得用链表了
    int cnt;    //当前桶内存放数据的个数
}Bucket;


void bucket_sort(int arr[],size_t len){
    int min = arr[0];
    int max = arr[0];
    size_t i;
    for(i=0;i<len;i++){
        if(min>arr[i]){    //找到最小值
            min = arr[i];    
        }
        if(max<arr[i]){    //找到最大值
            max = arr[i];    
        }
    }
    int size = max-min+1;
    Bucket bucket[5] = {};    //其实桶可以动态规划,但为了方便我这里直接分为5个桶
    for(i=0;i<len;i++){    //遍历待排序的数组,把每个元素放到相应的桶当中,
    //比如[0,200]之间的元素放到下标为0的桶中,[201,400]之间的元素放到下标为1的桶中..
    //以此类推,直到放完所有的数据
        int index = (arr[i]-min)/(size/5);    //用来判断当前元素arr[i]需要放到哪个桶当中
        bucket[index].vect[bucket[index].cnt++] = arr[i];
    }
    size_t j=0,k=0;
    for(i=0;i<5;i++){    //对这五个桶进行遍历
        count_sort(bucket[i].vect,bucket[i].cnt);    //首先对这个桶内的元素进行排序,
        //这里可以调用其他排序方法,也可以递归调用当前排序方法,但是为了节省内存,我选择调用其他排序方法,
        for(j=0;j<bucket[i].cnt;j++){
            arr[k++] = bucket[i].vect[j];    //对排序好的桶进行遍历,并且把里面的元素复制到arr中去    
        }
    }
}

12.基数排序

基数排序(radix sort)属于“分配式排序”(distribution sort),又称“桶子法”(bucket sort)或bin sort,顾名思义,它是透过键值的部份资讯,将要排序的元素分配至某些“桶”中,藉以达到排序的作用,基数排序法是属于稳定性的排序,其时间复杂度为O (nlog®m),其中r为所采取的基数,而m为堆数,在某些时候,基数排序法的效率高于其它的稳定性排序法。
解法:
1.首先根据个位数的数值,在走访数值时将它们分配至编号0到9的桶子中;
2.接下来将这些桶子中的数值重新串接起来,接着再进行一次分配,这次是根据十位数来分配;
3.接下来将这些桶子中的数值重新串接起来,持续进行以上的动作直至最高位数为止。
时间复杂度:设待排序列为n个记录,d个关键码,关键码的取值范围为radix,则进行链式基数排序的时间复杂度为O(d(n+radix)),其中,一趟分配时间复杂度为O(n),一趟收集时间复杂度为O(radix),共进行d趟分配和收集。
稳定性:稳定的算法;
代码实现:
还是定义桶的类型:

typedef struct Bucket{
    int vect[100];    //同样的可以用链表
    int cnt;
}Bucket;


void base_sort(int arr[],size_t len){
    size_t i;
    Bucket bucket[10] = {};    //十个桶
    int max = arr[0];
    for(i=0;i<len;i++){    //寻找最大值,就可以判断最大值的位数
        if(arr[i]>max){
            max = arr[i];    
        }    
    }
    size_t j,k;
    int num = 1;    //用来获得相应位数上的数字的关键参数,
    //比如要获得个位上的参数时num = 1;
    //获得十位上的数字时num = 10;
    //以此类推
    do{
        for(i=0;i<len;i++){    //遍历待排序的数组,把每个元素放入相应的桶中
        //比如251,当获得个位上的数字时,251放到下标为1的桶当中
        //当获得十位上的数字时,251放到下标为5的桶当中
        //当获得百位上的数字时,251放到下标为2的桶当中
        //当获得千位上的数字时,251放到下标为0的桶当中
        //以此类推
            int index = arr[i]/num%10;    //获得相应位数上的数字
            bucket[index].vect[bucket[index].cnt++] = arr[i];    //把这个数字放到相应的桶中
        }
        k=0;
        for(i=0;i<10;i++){
            for(j=0;j<bucket[i].cnt;j++){
                arr[k++] = bucket[i].vect[j];    //把这些桶按顺序依次遍历,
                //把桶中的元素重新放回arr当中
            }    
            bucket[i].cnt = 0;    //记得让桶中的cnt变为0,方便下一次存放
        }
        num*=10;    //num*10
    }while(max/=10);//循环条件
}

到此这篇关于C语言完整实现12种排序算法(小结)的文章就介绍到这了,更多相关C语言 排序算法内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C语言完整实现12种排序算法(小结)

本文链接: https://lsjlt.com/news/148883.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C语言完整实现12种排序算法(小结)
    目录1.冒泡排序2.插入排序3.折半插入排序4.希尔排序5.选择排序6.鸡尾酒排序7.堆排序8.快速排序9.归并排序10.计数排序11.桶排序12.基数排序1.冒泡排序 思路:比较相...
    99+
    2024-04-02
  • C语言怎么实现12种排序算法
    这篇文章主要介绍了C语言怎么实现12种排序算法的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C语言怎么实现12种排序算法文章都会有所收获,下面我们一起来看看吧。1.冒泡排序思路:比较相邻的两个数字,如果前一个数...
    99+
    2023-06-30
  • c语言实现的几种常用排序算法
    概述 最近重新回顾了一下数据结构和算法的一些基本知识,对几种排序算法有了更多的理解,也趁此机会通过博客做一个总结。 1.选择排序-简单选择排序 选择排序是最简单的一种基于O(n2)时...
    99+
    2024-04-02
  • C语言实现扫雷小游戏完整算法详解(附完整代码)
    目录前言1.算法基本思路2.算法详解1.初始化数组与打印数组2.设置雷3.排查与标记4.CountMine函数计算周围雷的个数 5.ExpandMine函数递归展开周围所有...
    99+
    2024-04-02
  • c语言如何实现排序算法
    小编给大家分享一下c语言如何实现排序算法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1.选择排序-简单选择排序选择排序是最简单的一种基于O(n2)时间复杂度的排...
    99+
    2023-06-15
  • java 算法 6种排序小结
    目录冒泡排序选择排序插入排序希尔排序归并排序快速排序冒泡排序 package 冒泡排序; import java.util.Arrays; public class Bub...
    99+
    2024-04-02
  • c语言入门2(12以内整数排序)
    #include <stdio.h> int main() {int a[11],i,j,t;//声明变量for(i=0;i<=10;i++)//初始化a[i]=0;for...
    99+
    2023-06-03
  • C语言实现快速排序算法实例
    首先我们要对一组数据进行排序: 在数组中选一个基准数(通常为数组第一个,黄圈圈标记了); 将数组中小于基准数的数据移到基准数左边,大于基准数的移到右边,怎么移动,后面说; 对于基准数...
    99+
    2024-04-02
  • C语言如何实现快速排序算法
    这篇文章将为大家详细讲解有关C语言如何实现快速排序算法,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。代码#define  _CRT_SECURE_NO_WARNINGS 1/...
    99+
    2023-06-22
  • C语言如何实现交换排序算法
    这篇文章主要介绍了C语言如何实现交换排序算法的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C语言如何实现交换排序算法文章都会有所收获,下面我们一起来看看吧。一、冒泡排序1.基本思想对于很多同学来说冒泡排序是再熟...
    99+
    2023-07-02
  • C语言排序算法实例分析
    这篇文章主要讲解了“C语言排序算法实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言排序算法实例分析”吧!1、直接插入排序基本思想:当插入第i(i>=1)个元素时,前面的ar...
    99+
    2023-06-29
  • C语言中怎样实现一个排序算法
    本篇文章给大家分享的是有关C语言中怎样实现一个排序算法,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。冒泡排序  冒泡排序(英语:BubbleS...
    99+
    2024-04-02
  • C语言常见排序算法归并排序
    目录前言 一、归并排序1.1 基本思想1.2 算法思想1.3 程序设计思想1.4 程序实现1.5 归并排序的特性总结前言 本期为大家带来的是常见排序算法中的归并排序,博主在...
    99+
    2024-04-02
  • C语言数据结构与算法之排序总结(一)
    目录一、前言二、基本概念1.排序2.排序方法的稳定性3.内部和外部排序三、插入类排序1.直接插入排序2.折半插入排序3.希尔排序四、交换类排序1.冒泡排序2.快速排序五、总结比较一、...
    99+
    2024-04-02
  • C语言数据结构与算法之排序总结(二)
    目录一、前言二、选择类排序1.简单选择排序2.树形选择排序3.堆选择排序三、归并排序四、分配类排序1.多关键字排序2.链式基数排序五、总结归纳一、前言 之前的排序总结(一)对插入类和...
    99+
    2024-04-02
  • Java 语言实现归并排序算法
    【引言】 归并排序算法是一种高效且稳定的排序算法。它采用分治法的思想,将数组反复分割成两个子数组,直到每个子数组只有一个元素。然后将这些子数组逐个合并,最终得到排序完毕的数组。本文将使用Java语言实现归并排序算法,并详细讲解其核心思想和代...
    99+
    2023-08-30
    排序算法 java 算法
  • C语言实现单链表的快速排序算法
    目录背景设计思路算法主要步骤快速排序算法实现整个程序源代码测试案例总结背景 传统QuickSort算法最大不足之处在于,由于其基于可索引存储结构设计(一般为数组或索引表),因而无法用...
    99+
    2024-04-02
  • 详细总结各种排序算法(Java实现)
    一、插入类排序1.直接插入排序思想:将第i个插入到前i-1个中的适当位置时间复杂度:T(n) = O(n²)。空间复杂度:S(n) = O(1)。稳定性:稳定排序。如果碰见一个和插入元素相等的,那么插入元素把想插入的元素放在相等元...
    99+
    2023-05-31
    java 排序算法 ava
  • python常用的各种排序算法原理与实现方法小结
    1. 冒泡排序(Bubble Sort) 基本思想:重复地遍历待排序的数列,每次比较相邻的两个元素,如果它们的顺序错误就交换位置,直到没有需要交换的元素为止。 实现代码: def b...
    99+
    2023-05-17
    python 排序算法
  • C语言实现二叉搜索树的完整总结
    目录1、 二叉树的构建2、二叉树的遍历前序遍历中序遍历后序遍历层序遍历4、二叉树的高度5、二叉树的删除6、由几种遍历序列还原二叉树 前序序列、中序序列还原二叉树:中序序列、...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作