举例讲解C语言对归并排序算法的基础使用
作者:飞翔的猫咪 发布时间:2021-05-24 19:48:09
标签:C语言,归并排序
基础概念
百度百科是这么描述归并排序的:
归并操作(merge),也叫归并算法,指的是将两个已经排序的序列合并成一个序列的操作。
设有数列
{6,202,100,301,38,8,1}
初始状态:
[6] [202] [100] [301] [38] [8] [1]
比较次数
i=1 [6 202 ] [ 100 301] [ 8 38] [ 1 ]3
i=2 [ 6 100 202 301 ] [ 1 8 38 ]4
i=3[ 1 6 8 38 100 202 301 ] 4
总计:11次
实例
#include <stdio.h>
void printArr(int arr[],int length){
int i;
for(i=0;i<length;i++){
printf("%d,",arr[i]);
}
printf("\n");
}
void merge(int a[],int alength,int b[],int blength,int c[]){//将2个已排好序的数组合并到数组c
int i=0,j=0,k=0;
while(1){
if(a[i]<=b[j]){
c[k] = a[i];
i++;
k++;
if(i==alength){
for(;j<blength;j++,k++){
c[k] = b[j];
}
break;
}
}else{
c[k] = b[j];
j++;
k++;
if(j==blength){
for(;i<alength;i++,k++){
c[k] = a[i];
}
break;
}
}
}
printArr(c,k);
}
void mergeSort(int arr[],int length){//将一个数组分成2个数组,前length-1为第一个,最后一个为第二个,然后合并2个数组
if(length > 1){
int arr1[length-1],arr2[1] = {arr[length-1]};
int i;
for(i=0;i<length-1;i++){
arr1[i] = arr[i];
}
mergeSort(arr1,length-1);//递归的调用自己
merge(arr1,length-1,arr2,1,arr);
}
}
int main(void){
int a[10] = {3,54,16,8,123,8,89,23,87,2};
printArr(a,10);
mergeSort(a,10);
return 0;
}
算法性能/复杂度
归并排序的效率是很高的,由于递归划分为子序列只需要logN复杂度,而合并每两个子序列需要大约2n次赋值,为O(n)复杂度,因此,只需要简单相乘即可得到归并排序的时间复杂度 O(㏒n)。并且由于归并算法是固定的,不受输入数据影响,所以它在最好、最坏、平均情况下表现几乎相同,均为O(㏒n)。
但是,归并排序最大的缺陷在于其空间复杂度。从上面的代码可以看到,在合并子数组的时候需要一个辅助数组,然后再把这个数据拷贝回原数组。所以,归并排序的空间复杂度(额外空间)为O(n)。可不可以省略这个数组呢?不行!如果取消辅助数组而又要保证原来的数组中数据不被覆盖,那就必须要在数组中花费大量时间来移动数据。不仅容易出错,还降低了效率。因此这个辅助空间是少不掉的。
算法稳定性
因为我们在遇到相等的数据的时候必然是按顺序“抄写”到辅助数组上的,所以,归并排序同样是稳定算法。
算法适用场景
归并排序在数据量比较大的时候也有较为出色的表现(效率上),但是,其空间复杂度O(n)使得在数据量特别大的时候(例如,1千万数据)几乎不可接受。而且,考虑到有的机器内存本身就比较小,因此,采用归并排序一定要注意。
0
投稿
猜你喜欢
- using Word;下面的例子中包括C#对Word文档的创建、插入表格、设置样式等操作:(例子中代码有些涉及数据信息部分被省略,重要是介绍
- Linux下JDK安装教程,具体内容如下1、下载 JDK Linux 版本(注意看自己安装 Linux 系统的位数)oracle 官网下载地
- 本文实例讲述了C#读取文本文件到listbox组件的方法。分享给大家供大家参考。具体实现方法如下:private void AddTxtTo
- 很久之前也写过一篇使用Jitpack发布Android开源库的文章,详见Android发布项目到jitpack的完整步骤近来因为工作原因,又
- 本文实例讲述了Java编程实现基于TCP协议的Socket聊天室。分享给大家供大家参考,具体如下:这里使用Socket套接字进行编程,完成的
- 本文实例为大家分享了java实现简单俄罗斯方块的具体代码,供大家参考,具体内容如下结合网上的资料刚做完课程设计,具体代码如下:public
- 本文实例讲述了C#使用Ado.Net更新和添加数据到Excel表格的方法。分享给大家供大家参考。具体分析如下:微软NET提供了一个交互的方法
- 冒泡排序在八大排序中,冒泡排序是最为出名的排序算法之一!冒泡排序的代码还是相当简单的,两层循环,外层是冒泡轮数,里层是依次比较,这个算法的时
- 本文实例为大家分享了Android学习之Broadcast的使用方法,供大家参考,具体内容如下实现开机启动提示网络的广播package co
- 1. Resource Basics(1) Manifest Resources(资源清单)资源在编译期间添加到程序集。如果要将资源嵌入到程
- 该着色方法一句着 * 层中要素类的某个数值字段的属性值,按这个属性值为每种不同值得要素单独分配一种显示符号样式。关键在于获取该字段所有要素的唯
- 本文实例为大家分享了C# this关键字的四种用法,供大家参考,具体内容如下用法一 this代表当前实例,用this.显式调用一
- Java 分割字符串java.lang.String 的 split() 方法, JDK 1.4 or laterpublic
- 项目配置依赖首先搭建一个标准的SpringBoot项目工程,相关版本以及依赖如下本项目借助SpringBoot 2.2.1.RELEASE
- 背包问题主要是指一个给定容量的背包、若干具有一定价值和重量的物品,如何选择物品放入背包使物品的价值最大。其中又分01背包和无限背包,这里主要
- 一、前言什么是多渠道打包以及多渠道打包可以做什么,这里就不做介绍了,相信看到这篇文章的你已经了解了,多渠道打包的方式比较多,这里我们用Gra
- Android init.rc文件详解本文主要来自$ANDROID_SOURCE/system/init/readme.txt的翻译.1 简
- 在JDK的Collection中我们时常会看到类似于这样的话:例如,ArrayList:注意,迭代器的快速失败行为无法得到保证,因为一般来说
- 一、Servlet3.0异步请求@WebServlet(value = "/async", asyncSupported
- 摘要:其实两种方法归结起来看还是一种,都是利用Thread的构造器进行创建,区别就是一种是无参的,一种是有参的。一、继承Thread线程类: