首页 > 编程语言 >C++编程-数据排序2

C++编程-数据排序2

时间:2024-08-25 19:51:35浏览次数:15  
标签:归并 int 编程 C++ 冒泡排序 有序 排序 逆序

关于以后的更新

已经8月25号了,即将接近CSP-J/S,因此,在数据排序算法更新完后,我们会重点更新CSP的试卷以及知识点,希望大家在考试中旗开得胜!

回顾数据排序1

在数据排序1中,我们讲解了选择、冒泡、插入、桶、快速排序,并留下了2道题目,今天就来解答这两道题目

一:冒泡排序

#include <stdio.h>
int main() {
	int n, i, j, temp;
	int a[200];
	scanf("%d", &n);
	for (i = 0;i < n;i++)
		scanf("%d", &a[i]);
	for (i = 0;i < n;i++) {
		for (j = 0;j + i + 1 < n;j++) {
			if (a[j] > a[j + 1]) {
				temp = a[j];
				a[j] = a[j + 1];
				a[j + 1] = temp;
			}
		}
	}
	for (i = 0;i < n;i++) {
		printf("%d ", a[i]);
	}
	puts("");
	return 0;
}

二:选择排序

#include <iostream>
#include <algorithm>
using namespace std;
int a[15];
int main()
{
	for (int i=0;i<10;i++)
		cin>>a[i];
	sort(a+0,a+10);
	for (int i=0;i<10;i++)
		cout <<a[i]<<endl;
	return 0;
} 

今日讲解

在本期中,我们需要讲解归并、逆序对排序的方法,还要学习各种排序算法之间的比较,那么开始吧!

例题六:归并排序

算法简介

归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

      例如有8个数据需要排序:10 4 6 3 8 2 5 7 

      归并排序主要分两大步:分解、合并。

      合并过程为:比较a[i]和a[j]的大小,若a[i]≤a[j],则将第一个有序表中的元素a[i]复制到r[k]中,并令i和k分别加上1;否则将第二个有序表中的元素a[j]复制到r[k]中,并令j和k分别加上1,如此循环下去,直到其中一个有序表取完,然后再将另一个有序表中剩余的元素复制到r中从下标k到下标t的单元。归并排序的算法我们通常用递归实现,先把待排序区间[s,t]以中点二分,接着把左边子区间排序,再把右边子区间排序,最后把左区间和右区间用一次归并操作合并成有序的区间[s,t]。

题目描述

用归并排序,完成归并排序的子函数

标准程序

void msort(int s,int t)
{
    if(s==t) return;                 //如果只有一个数字则返回,无须排序
    int mid=(s+t)/2;  
    msort(s,mid);                  //分解左序列
    msort(mid+1,t);               //分解右序列
    int i=s, j=mid+1, k=s;      //接下来合并
    while(i<=mid && j<=t)
    {
          if(a[i]<=a[j])  
          {
                r[k]=a[i]; k++; i++;
           }else{
                        r[k]=a[j]; k++; j++;
                    }
      }  
      while(i<=mid)               //复制左边子序列剩余
      {
           r[k]=a[i]; k++; i++;
       }
       while(j<=t)                    //复制右边子序列剩余  
       {
            r[k]=a[j]; k++; j++;
        }
        for(int i=s; i<=t; i++) a[i]=r[i];  
        return 0;
} 

标程分析

归并排序的时间复杂度是O(nlogn),速度快。同时,归并排序是稳定的排序。即相等的元素的顺序不会改变。如输入记录 1(1) 3(2) 2(3) 2(4) 5(5) (括号中是记录的关键字)时输出的 1(1) 2(3) 2(4) 3(2) 5(5) 中的2 和 2 是按输入的顺序。这对要排序数据包含多个信息而要按其中的某一个信息排序,要求其它信息尽量按输入的顺序排列时很重要,这也是它比快速排序优势的地方。

例题七:逆序对

算法简介

       上述提到归并排序是稳定的排序,相等的元素的顺序不会改变,进而用其可以解决逆序对的问题。首先我们了解一下什么是逆序对。

       逆序对:设 A 为一个有 n 个数字的有序集 (n>1),其中所有数字各不相同。如果存在正整数 i, j 使得 1 ≤ i < j ≤ n 而且 A[i] > A[j],则 <A[i], A[j]> 这个有序对称为 A 的一个逆序对,也称作逆序数。

        例如,数组(3,1,4,5,2)的逆序对有(3,1),(3,2),(4,2),(5,2),共4个。

        所谓逆序对的问题,即对给定的数组序列,求其逆序对的数量。

        从逆序对定义上分析,逆序对就是数列中任意两个数满足大的在前,小的在后的组合。如果将这些逆序对都调整成顺序(小的在前,大的在后),那么整个数列就变的有序,即排序。因而,容易想到冒泡排序的机制正好是利用消除逆序来实现排序的,也就是说,交换相邻两个逆序数,最终实现整个序列有序,那么交换的次数即为逆序对的数量。

        冒泡排序可以解决逆序对问题,但是由于冒泡排序本身效率不高,时间复杂度为O(n2),对于n比较大的情况就没用武之地了。我们可以这样认为,冒泡排序求逆序对效率之所以低,是因为其在统计逆序对数量的时候是一对一对统计的,而对于范围为n的序列,逆序对数量最大可以是(n+1)*n/2,因此其效率太低。那怎样可以一下子统计多个,而不是一个一个累加呢?这个时候,归并排序就可以帮我们来解决这个问题。

       在合并操作中,我们假设左右两个区间元素为:

       左边:{3 4 7 9}                右边:{1 5 8 10}

       那么合并操作的第一步就是比较3和1,然后将1取出来,放到辅助数组中,这个时候我们发现,右边的区间如果是当前比较的较小值,那么其会与左边剩余的数字产生逆序关系,也就是说1和3、4、7、9都产生了逆序关系,我们可以一下子统计出有4对逆序对。接下来3,4取下来放到辅助数组后,5与左边剩下的7、9产生了逆序关系,我们可以统计出2对。依此类推,8与9产生1对,那么总共有4+2+1对。这样统计的效率就会大大提高,便可较好的解决逆序对问题。

      而在算法的实现中,我们只需略微修改原有归并排序,当右边序列的元素为较小值时,就统计其产生的逆序对数量,即可完成逆序对的统计。

题目描述

利用逆序对算法,完成子函数

标准程序

void msort(int s,int t)
{
    if(s==t) return;                  //如果只有一个数字则返回,无须排序
    int mid=(s+t)/2;
    msort(s,mid);                   //分解左序列
    msort(mid+1,t);                //分解右序列
    int i=s, j=mid+1, k=s;       //接下来合并
    while(i<=mid && j<=t)
    {
        if(a[i]<=a[j]) 
        {     r[k]=a[i]; k++; i++;
        }else{
                    r[k]=a[j]; k++; j++;  
                     ans+=mid-i+1;  //统计产生逆序对的数量
                 }
    }
    while(i<=mid)                    //复制左边子序列剩余
    {
         r[k]=a[i]; k++; i++;
    }
    while(j<=t)                         //复制右边子序列剩余
    {
         r[k]=a[j]; k++; j++;
    }
    for(int i=s; i<=t; i++) a[i]=r[i];
    return 0;
}

标程分析

其中ans+=mid-i+1这句代码统计新增逆序对的数量,ans作为全局变量,用于统计逆序对的数量,此时ans要增加左边区间剩余元素的个数。当归并排序结束后,逆序对问题也得到解决,ans即为逆序对的数量。

本期重点:各种排序算法的比较

先言!!!

建议拿好笔记本,记好关键字。

1.稳定性比较

       插入排序、冒泡排序、二叉树排序、二路归并排序及其他线形排序是稳定的。

       选择排序、希尔排序、快速排序、堆排序是不稳定的。

2.时间复杂度比较

       插入排序、冒泡排序、选择排序的时间复杂性为O(n2);快速排序、堆排序、归并排序的时间复杂性为O(nlog2n);桶排序的时间复杂性为O(n);

       若从最好情况考虑,则直接插入排序和冒泡排序的时间复杂度最好,为O(n),其它算法的最好情况同平均情况相同;若从最坏情况考虑,则快速排序的时间复杂度为O(n2),直接插入排序和冒泡排序虽然平均情况相同,但系数大约增加一倍,所以运行速度将降低一半,最坏情况对直接选择排序、堆排序和归并排序影响不大。

       由此可知,在最好情况下,直接插入排序和冒泡排序最快;在平均情况下,快速排序最快;在最坏情况下,堆排序和归并排序最快。

3.辅助空间的比较

       桶排序、二路归并排序的辅助空间为O(n),快速排序的辅助空间为O(log2n),最坏情况为O(n),其它排序的辅助空间为O(1);

4.其他

       插入、冒泡排序的速度较慢,但参加排序的序列局部或整体有序时,这种排序能达到较快的速度。反而在这种情况下,快速排序反而慢了。

       当n较小时,对稳定性不作要求时宜用选择排序,对稳定性有要求时宜用插入或冒泡排序。

       若待排序的记录的关键字在一个明显有限范围内时,且空间允许是用桶排序。

       当n较大时,关键字元素比较随机,对稳定性没要求宜用快速排序。

       当n较大时,关键字元素可能出现本身是有序的,对稳定性没有要求时宜用堆排序

    快速排序是目前基于比较的内部排序中被认为是最好的方法,当待排序的关键字是随机分布时,快速排序的平均时间最短;

堆排序所需的辅助空间少于快速排序,并且不会出现快速排序可能出现的最坏情况。这两种排序都是不稳定的。

小练习

今天就讲这么多,下面来一道练习题

要选择合适的方法啊(别™使用sort)

排序

题目描述

题目描述

输入三个整数,按由小到大的顺序输出。

输入

三个整数

输出

由小到大输出成一行,每个数字后面跟一个空格

样例输入 复制
2 3 1
样例输出 复制
1 2 3 

答案在下期公布,下期还需要讲一些数据排序的一些知识点,下下期就得将CSP的内容了

标签:归并,int,编程,C++,冒泡排序,有序,排序,逆序
From: https://blog.csdn.net/weixin_68261272/article/details/141533669

相关文章

  • 【面试系列】30个常见的初级SQL编程题
    欢迎来到我的博客,很高兴能够在这里和您见面!欢迎订阅相关专栏:工......
  • C/C++、Java、Python全面比较
    以下是对C/C++、Java、Python三种编程语言的全面比较,以表格形式呈现:特性/语言C/C++JavaPython类型系统静态类型静态类型动态类型内存管理手动管理自动管理(垃圾回收)自动管理(垃圾回收)编译/解释编译型编译型(通过JVM解释执行)解释型性能高(直接编译成机器码)中等(通过JIT优化)较低(解......
  • 【编程基础】亲密数对
    题目描述键盘输入N,N在2至2000之间,求2至N中的亲密数对,就是A的因子和等于B,B的因子和等于A,且A≠B。如48和75是亲密数对。48的因子和为2+3+4+6+8+12+16+24=75,而75的因子和为3+5+15+25=48。输入只有一行,为一个整数N(2<=N<=2000)输出输出若干行,每行两个整数(用一个空格隔开)。样......
  • 从零开始学习C++之枚举与模拟
    枚举和模拟是C++中最为基础的算法,也是之后赛时部分分的算法首选。枚举顾名思义,枚举就是将所有值全部扫一遍。枚举算法的流程图如下:我们很容易就可以写出伪代码:for(枚举区间){ 代码,例: if(条件) { 输出 }}模拟模拟就是将做的事情通过程序一步步完成,有时候很简......
  • 除Qt以外的C++GUI库
    ImGui图形用户界面项目Github地址:https://github.com/ocornut/imguiwxWidgetsHome:https://wxwidgets.org/。NanoGUINanoGUI是用于OpenGL3+、GLES2/3和Metal的极简跨平台工具库。RmlUiRmlUi是基于HTML和CSS标准的C++GUI库,目标是为任何项目的界面需求提供完整的解决......
  • [C++] 异常详解
    标题:[C++]异常详解@水墨不写bug目录一、错误处理方式C语言Java语言二、异常的概念三、异常的使用1.异常的抛出和捕获(基本用法) 2.异常的重新抛出(特殊情况)3.异常的规范和常见坑点四、标准库的异常体系五、C++异常小结正文开始:一、错误处理方式   ......
  • [C++] 初识 智能指针
    标题:[C++]初识智能指针@水墨不写bug目录一、前言二、智能指针1.什么是RAII?2.智能指针分类 三、智能指针简介1.std::auto_ptr2.std::unique_ptr3.std::shared_ptr正文开始:一、前言    C++智能指针的出现是有一定的背景的:    Java有专属......
  • 莫队算法C/C++实现
    目录简介 算法原理算法步骤C++实现应用场景莫队算法(Mo'sAlgorithm)是一种用于解决区间查询和更新问题的算法,由俄罗斯选手莫洛佐夫(MoMorozov)提出。它在算法竞赛和某些计算密集型任务中非常有用,尤其是在需要处理大量区间查询和更新操作时。莫队算法以其高效性和简洁性......
  • A*算法C/C++实现
    A*算法是一种在图形平面上,有多个节点的路径中,寻找一条从起始点(source)到目标点(goal)的最短遍历路径的算法。它属于启发式搜索算法,因为它使用启发式方法来计算图中的节点,从而减少实际计算的节点数量。A*(A星)算法是一种启发式搜索算法,用于在图中找到从起始点(source)到目标点(goal)的......
  • Java Stream:高效编程的利器与潜在陷阱
    Java8引入的StreamAPI为处理集合数据提供了一种全新的方式,使开发者能够以声明性风格进行操作。Stream流使得代码更加简洁优雅,同时也提高了并行处理的效率。然而,Stream流的使用也带来了一些潜在的缺点。本文将深入分析JavaStream流操作的优缺点。一、JavaStream流操作的优......