归并排序总结

1.归并排序

归并排序的步骤如下:

①枚举中点,将区间分为左右两段;

②对左右两段区间分别排序;

        这个过程以递归的方式进行。

③合并两段区间。

        是一个模拟的过程。用两个指针分别指向左右区间,判断当前哪个数小,将小的数合并进总区间,指针后移。当指针移到某个区间的末尾,则将另一个区间的剩余部分直接接到总区间后面。

代码模板如下:

#include<iostream>
using namespace std;
const int N=1e5+10;
int n,a[N],tmp[N];
void merge_sort(int a[],int l,int r)
{
    if(l>=r) return;
    int mid=(l+r)>>1;
    merge_sort(a,l,mid);
    merge_sort(a,mid+1,r);
    int i=l,j=mid+1,k=0;
    while(i<=mid&&j<=r)
    {
        if(a[j]<=a[i]) tmp[k++]=a[j++];
        else tmp[k++]=a[i++];
    }
    while(i<=mid) tmp[k++]=a[i++];
    while(j<=r) tmp[k++]=a[j++];
    for(int i=l,j=0;i<=r;i++,j++) a[i]=tmp[j];
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%d",&a[i]);
    merge_sort(a,1,n);
    for(int i=1;i<=n;i++) printf("%d ",a[i]);
}

2.求逆序对数量

归并排序一个很重要的作用就是用来求数组中的逆序对数量。

在归并排序过程中,当我们发现ai>aj时,所有[ai,mid]区间内的数都可以和aj构成逆序对,我们把这些数加起来,就能计算出所有的逆序对数量。

#include<iostream>
using namespace std;
const int N=1e5+10;
typedef long long ll;
int n,a[N],tmp[N];

ll merge_sort(int a[],int l,int r)
{
    if(l>=r) return 0;
    int mid=(l+r)>>1;
    ll res=merge_sort(a,l,mid)+merge_sort(a,mid+1,r);
    int k=0,i=l,j=mid+1;
    while(i<=mid&&j<=r)
    {
        if(a[i]>a[j])
        {
            res+=(mid-i+1);
            tmp[k++]=a[j++];
        }
        else tmp[k++]=a[i++];
    }
    while(i<=mid) tmp[k++]=a[i++];
    while(j<=r) tmp[k++]=a[j++];
    for(int i=l,j=0;i<=r;i++,j++) a[i]=tmp[j];
    return res;
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%d",&a[i]);
    ll res=merge_sort(a,1,n);
    printf("%lld",res);
}

3.小朋友排队

我们可以知道,通过交换,我们每次可以最多减少1个逆序对,假设逆序对数量为k,要完成排序,我们交换的总次数最少是k次,而这个最少次数是可以达到的(在冒泡排序中就可以达到),而且在达到这个次数时,我们的交换方案是恒定的(必须按照冒泡排序的方式)。

对于某个小朋友,如果他前面有k1个比他高的小朋友,后面有k2个比他低的小朋友,要使最后小朋友的身高变为从小到大排列,那么这个小朋友至少要被交换k1+k2次。

而k1+k2的值,我们可以借助归并排序求出。

当a[i]>a[j]时,[i,mid]范围内的数都大于a[j],对于a[j]来说,有mid-i+1个逆序对,即a[j]的k1=mid-i+1;

当a[j]>=a[i]时,[mid+1,j-1]范围内的数都小于a[i],对于a[i]来说,有j-mid-1个逆序对,即a[i]的k2=j-mid-1。

#include<iostream>
using namespace std;
const int N=1e5+10;
typedef pair<int,int> PII;
typedef long long ll;
#define x first
#define y second
int n;
PII a[N],tmp[N];
ll cnt[N];
void merge_sort(PII a[],int l,int r)
{
    if(l>=r) return;
    int mid=(l+r)>>1;
    merge_sort(a,l,mid);
    merge_sort(a,mid+1,r);
    int k=0,i=l,j=mid+1;
    while(i<=mid&&j<=r)
    {
        if(a[i].y>a[j].y) 
        {
            cnt[a[j].x]+=(mid-i+1);
            tmp[k++]=a[j++];
        }
        else
        {
            cnt[a[i].x]+=(j-mid-1);
            tmp[k++]=a[i++];
        }
    }
    while(i<=mid) 
    {
        cnt[a[i].x]+=(r-mid);
        tmp[k++]=a[i++];
    }
    while(j<=r) 
    {
        tmp[k++]=a[j++];
    }
    for(int i=l,j=0;i<=r;i++,j++) a[i]=tmp[j];
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++) 
    {
        a[i].x=i;
        scanf("%d",&a[i].y);
    }
    merge_sort(a,1,n);
    ll res=0;
    for(int i=1;i<=n;i++) res+=(cnt[i]+1)*cnt[i]/2;
    printf("%lld",res);
}

4.火柴排队

505. 火柴排队 - AcWing题库

首先需要一点点贪心,我们可以证明出,当a、b都单调排列时,得到的距离是最小的。证明的过程是反证法,我们假设a(i)<a(i+1),b(i)>b(i+1),与a(i)<a(i+1),b(i)<b(i+1)的情况进行比较就可以很容易证明。

接下来,我们可以将问题简化,假设a数组是单调递增的,那么此时的最小交换次数就等于b的逆序对数量。我自己在这里就犯了一个小错误,到这一步,就想当然地认为最后的结果就等于b数组逆序对数量-a数组逆序对数量。其实不是,逆序对数量相等的两个数组并不是完全相同的。

我们继续往下思考。其实,我们只需要保证a、b数组中按照从小到大排序,处于一个位置的数一一对应就好了。由此,我们可以将a数组的1~n位数字映射成1~n,并将b数组中的数字按照这个映射规则映射一遍,此时情况就变成了我们上一步思考的简化情况:a数组升序排列。而b数组的数,因为使用和a数组相同的字典也映射了一遍,因此可以反映出实际中a、b数组中应对应的数的位置差距。

在映射时感觉有一个小bug,我只用a中的元素构建了映射表,但是,如果b中出现了a中没有出现的元素,我的映射会出现问题。

#include<iostream>
#include<unordered_map>
#include<algorithm>
using namespace std;
const int N=1e5+10;
const int mod=99999997;
typedef long long ll;
int a[N],b[N],tmp[N],c[2*N];
int n;
unordered_map<int,int> ha;
ll merge_sort(int a[],int l,int r)
{
    if(l>=r) return 0;
    int mid=(l+r)>>1;
    ll res=merge_sort(a,l,mid)+merge_sort(a,mid+1,r);
    res%=mod;
    int k=0,i=l,j=mid+1;
    while(i<=mid&&j<=r)
    {
        if(a[i]>a[j])
        {
            res+=mid-i+1;
            tmp[k++]=a[j++];
        }
        else tmp[k++]=a[i++];
    }
    while(i<=mid) 
    {
        tmp[k++]=a[i++];
    }
    while(j<=r) tmp[k++]=a[j++];
    for(int i=l,j=0;i<=r;i++,j++) a[i]=tmp[j];
    return res%mod;
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%d",&a[i]);
    for(int j=1;j<=n;j++) scanf("%d",&b[j]);
    int k=1;
    
    for(int i=1;i<=n;i++)
    {
        if(ha.find(a[i])==ha.end()) ha.insert({a[i],k++});
    }
    
    for(int i=1;i<=n;i++) b[i]=ha[b[i]];
    ll k2=merge_sort(b,1,n);
    printf("%lld",k2);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/435510.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

FPGA——三速自适应以太网设计(2)GMII与RGMII接口

FPGA——以太网设计&#xff08;2&#xff09;GMII与RGMII 基础知识&#xff08;1&#xff09;GMII&#xff08;2&#xff09;RGMII&#xff08;3&#xff09;IDDR GMII设计转RGMII接口跨时钟传输模块 基础知识 &#xff08;1&#xff09;GMII GMII:发送端时钟由MAC端提供 下…

k近邻分类算法实现(KNN)

KNN算法实现 最近要用到对 某些数据进行自动识别分类&#xff0c;简单学习了一下k近邻算法&#xff0c;分享一下。 例如&#xff1a;电影动作片爱情片分类识别 这里我们使用了sklearn库&#xff0c;它用起来简单方便。 先提供代码如下&#xff1a; import numpy as np imp…

docker的简单使用

在一些进行使用靶场或者工具的时候&#xff0c;我们可以用docker在线拉取&#xff0c;就可以省去手动搭建靶场的过程 一、docker的配置 因为docker是默认从docker的官网进行拉取&#xff0c;所以拉取经常速度很慢或者失败&#xff0c;我们先要进行一下配置&#xff0c;让他优…

欧科云链:角力Web3.0,香港如何为合规设线?

在香港拥抱Web3.0的过程中,以欧科云链为代表的合规科技企业将凸显更大重要性。 ——据香港商报网报道 据香港明报、商报等媒体报道&#xff0c;港区全国政协兼香港选委界立法会议员吴杰庄在日前召开的全国两会上提出在大湾区建设国际中小企业创新Web3融资平台等提案&#xff0…

《系统架构设计师教程(第2版)》第6章-据库设计基础知识-01-数据库基本概念

文章目录 1. 概述1.1 基本概念1&#xff09;信息 (Information)2&#xff09;数据 (Data)3&#xff09;数据库 (DB)4&#xff09;数据库系统(DBS)5&#xff09;数据库管理系统&#xff08;DBMS&#xff09; 1.2 数据库技术的发展1.2.1 人工管理阶段1.2.2 文件系统阶段1&#xf…

C++中OpenMP的使用方法

适用场景 OpenMP是一种用于共享内存并行系统的多线程程序设计方案&#xff1b;简单地说&#xff0c;OpenMP通过一种较为简单的使用方式&#xff0c;实现代码的CPU并行化处理&#xff0c;从而最大化利用硬件的多核性能&#xff0c;成倍地提升处理效率&#xff1b; OpenMP适用场…

springboot3.x集成SpringDoc Swagger3

近期将springboox2.x升级到了3.x&#xff0c;索性将swagger2也同步升级到swagger3&#xff0c;具体过程如下。 一、添加maven依赖 <dependency><groupId>org.springdoc</groupId><artifactId>springdoc-openapi-starter-webmvc-ui</artifactId>…

每日力扣——滑动窗口与前 K 个高频元素

&#x1f525; 个人主页: 黑洞晓威 &#x1f600;你不必等到非常厉害&#xff0c;才敢开始&#xff0c;你需要开始&#xff0c;才会变的非常厉害。 滑动窗口最大值 给定一个数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑…

uni app 微信小程序微信支付

使用方法 接口传参 使用wx.requestPayment方法是一个统一各平台的客户端支付API&#xff0c;不管是在某家小程序还是在App中&#xff0c;客户端均使用本API调用支付

STM32自学☞WDG(看门狗)及其案例

一、WDG简介 由于看门狗的代码很少所以就直接在main主函数中写了&#xff0c;没单独建文件 二、独立看门狗 涉及的按键可参考之前的key.c和key.h文件 独立看门狗配置流程&#xff1a; 1.开启时钟&#xff08;LSI&#xff09; 2.解除IWDG_PR和IWDG_RLR的写保护 3.写入预分频和重…

[HackMyVM]靶场 Wild

kali:192.168.56.104 主机发现 arp-scan -l # arp-scan -l Interface: eth0, type: EN10MB, MAC: 00:0c:29:d2:e0:49, IPv4: 192.168.56.104 Starting arp-scan 1.10.0 with 256 hosts (https://github.com/royhills/arp-scan) 192.168.56.1 0a:00:27:00:00:05 …

Golang的Channel源码阅读、工作流程分析。

Channel整体结构 源码位置 位于src/runtime下的chan.go中。 Channel整体结构图 图源&#xff1a;https://i6448038.github.io/2019/04/11/go-channel/ Channel结构体 type hchan struct {qcount uint // total data in the queuedataqsiz uint // si…

python unittest实现api自动化测试

这篇文章主要为大家详细介绍了python unittest实现api自动化测试的方法&#xff0c;具有一定的参考价值&#xff0c;感兴趣的小伙伴们可以参考一下 项目测试对于一个项目的重要性&#xff0c;大家应该都知道吧&#xff0c;写python的朋友&#xff0c;应该都写过自动化测试脚本…

Nginx启动服务

Nginx启动服务 一、启动前置 下载地址 如已安装Docker&#xff0c;下一步拉取Nginx最新的Docker镜像&#xff1a; docker pull nginx:latest查看拉取下来的镜像&#xff1a; docker images二、启动服务 创建Docker容器&#xff1a; docker run --name {projectname} -p 80…

Open3D 生成空间3D椭圆点云

目录 一、算法原理二、代码实现三、结果展示本文由CSDN点云侠原创,原文链接。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。 一、算法原理 设椭圆在 X O Y XOY XO

MySQL基础-----SQL语句之DCL数据控制语句

目录 前言 一、管理用户 1.查询用户 2.创建用户 3.修改用户密码 4.删除用户 案例 二、权限控制 1.查询权限 2.授予权限 3.撤销权限 案例 前言 本期我们学习SQL语句的最后一部分内容&#xff0c;也就是数据控制语句DCL。DCL英文全称是Data Control Language(数据控制语…

支部管理系统微信小程序(管理端+用户端)flask+vue+mysql+微信小程序

系统架构如图所示 高校D支部管理系统 由web端和微信小程序端组成&#xff0c;由web端负责管理&#xff0c;能够收缴费用、发布信息、发布问卷、发布通知等功能 部分功能页面如图所示 微信小程序端 包含所有源码和远程部署&#xff0c;可作为毕设课设

ctf_show笔记篇(web入门---文件上传)

文件上传 151&#xff1a;简单的前端验证&#xff0c;有多种绕过方法 152&#xff1a;简单后端验证&#xff0c;不知道过滤了那些后缀&#xff0c;我尝试以后都可以上传 153&#xff1a;利用.user.ini文件&#xff0c;虽然能上传.pht这一类文件但访问时只会下载下来 这里就…

本地项目推送到腾讯云轻量应用服务器教程(并实现本地推送远程自动更新)

将本地项目上传到腾讯云轻量应用服务器并实现后续的推送更新&#xff0c;具体步骤如下&#xff1a; 在本地项目目录下初始化 Git 仓库&#xff1a; cd 项目目录 git init将项目文件添加到 Git 仓库并提交&#xff1a; git add . git commit -m "Initial commit"在…

【unity实战】3D水系统,游泳,潜水,钓鱼功能实现

文章目录 素材将项目升级为URP画一个水潭地形材质升级为URP创建水调节水第一人称人物移动控制游泳水面停留添加水下后处理水下呼吸钓鱼参考完结 素材 https://assetstore.unity.com/packages/vfx/shaders/urp-stylized-water-shader-proto-series-187485 将项目升级为URP 这…