归并排序总结

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);
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于机器学习的垃圾分类

1绪论 1.1问题背景 垃圾分类有减少环境污染、节省土地资源、再生资源的利用、提高民众价值观念等的好处&#xff0c;在倡导绿色生活&#xff0c;注重环境保护的今天&#xff0c;正确的垃圾分类和处理对我们的生态环境显得尤为重要。 在国外很多国家&#xff0c;经过了几十年…

VTK的编译和部署,配合c++和visual studio2022,VTK开发环境的配置

1.下载 在官网选择最新的版本 Download | VTK 下载之后进行解压&#xff0c;然后再里面创建build目录&#xff0c;方便后面使用cmake进行编译 2.对源码进行编译 打卡Cmake&#xff0c;如图操作 可以看到点击configure之后&#xff0c;cmake对我们的代码在进行处理 处理完成之…

基于SpringBoot+Vue+ElementUI+Mybatis前后端分离管理系统超详细教程(二)

学习后端CRUD操作 书接上文&#xff0c;我们学习了前后端分离项目的基础环境配置和用户管理模块的前后端基础搭建&#xff0c;以下链接是上一节教程内容详细步骤&#xff0c;友友们可以跟着步骤实操。本节课程我们在前面项目的基础上接着学习后端CRUD操作&#xff0c;真正打通数…

阿里云一键登录(号码认证服务)

前言 用户登录原来的登录方式如下 1. 手机号验证码 2. 账号密码 运营觉得操作过于复杂, 因此想引入阿里自动登录的逻辑, 也就是号码认证服务,所以才有了这篇问文章 注: 本文只是记录Java端的实现, app端的请自行查询文档实现 官方资料 文档 : 什么是号码认证服务_号码认证服务(…

SpringBoot中集成LiteFlow(轻量、快速、稳定可编排的组件式规则引擎)实现复杂业务解耦、动态编排、高可扩展

场景 在业务开发中&#xff0c;经常遇到一些串行或者并行的业务流程问题&#xff0c;而业务之间不必存在相关性。 使用策略和模板模式的结合可以解决这个问题&#xff0c;但是使用编码的方式会使得文件太多, 在业务的部分环节可以这样操作&#xff0c;在项目角度就无法一眼洞…

【洛谷 P9240】[蓝桥杯 2023 省 B] 冶炼金属 题解(二分答案)

[蓝桥杯 2023 省 B] 冶炼金属 题目描述 小蓝有一个神奇的炉子用于将普通金属 O 冶炼成为一种特殊金属 X。这个炉子有一个称作转换率的属性 V V V&#xff0c; V V V 是一个正整数&#xff0c;这意味着消耗 V V V 个普通金属 O 恰好可以冶炼出一个特殊金属 X&#xff0c;当普…

产业园区如何实现数字化运营管理?

​在数字化浪潮席卷全球的今天&#xff0c;产业园区正经历着前所未有的变革&#xff0c;数字化运营管理成为各个园区转型升级的发力方向&#xff0c;它不仅能够提升园区的运营管理效率&#xff0c;还能够帮助园区提高服务效能、实现精准招商、增强决策效率&#xff0c;从而全面…

Redis实战—商户查询缓存

本博客为个人学习笔记&#xff0c;学习网站&#xff1a;黑马程序员Redis入门到实战 实战篇之商户查询缓存 目录 什么是缓存 添加Redis缓存 缓存更新策略 数据库缓存不一致解决方案 案例&#xff1a;给查询商铺的缓存添加超时剔除和主动更新策略 缓存穿透 案例&#xff1…

奇富科技:大数据任务从诊断到自愈的实践之路

一、为什么要做诊断引擎 毓数平台是奇富科技公司自主研发的一站式大数据管理、开发、分析平台&#xff0c;覆盖大数据资产管理、数据开发及任务调度、自助分析及可视化、统一指标管理等多个数据生命周期流程&#xff0c;让用户使用数据的同时&#xff0c;挖掘数据最大的价值。…

打造高效、安全的交易平台:开发流程与关键要素解析

在数字化时代&#xff0c;大宗商品交易平台开发/搭建已成为连接买家与卖家的桥梁&#xff0c;为无数企业和个人提供了便捷、高效的交易机会。然而&#xff0c;随着市场的竞争日益激烈&#xff0c;如何打造一个既符合用户需求又具备竞争力的交易平台&#xff0c;成为了众多开发者…

文件上传{session文件包含以及条件竞争、图片文件渲染绕过(gif、png、jpg)}

session文件包含以及条件竞争 条件&#xff1a; 知道session文件存储在哪里 一般的默认位置&#xff1a; /var/lib/php/sess_PHPSESSID /var/lib/php/sessions/sess_PHPSESSID /tmp/sess_PHPSESSID /tmp/sessions/sess_PHPSESSID ####在没做过设置的情况下一般都是存储在/var…

解决WordPress更新插件或者更新版本报WordPress 需要访问您网页服务器的权限的问题

文章目录 前言一、原因二、解决步骤总结 前言 当对WordPress的插件或者版本进行更新时报错&#xff1a;要执行请求的操作&#xff0c;WordPress 需要访问您网页服务器的权限。 请输入您的 FTP 登录凭据以继续。 如果您忘记了您的登录凭据&#xff08;如用户名、密码&#xff09…

光线追踪7 - 抗锯齿(Antialiasing)

目前为止&#xff0c;如果你放大渲染出的图像&#xff0c;可能会注意到图像边缘的明显“阶梯状”效果。这种阶梯效果通常被称为“走样”或“锯齿”。当真实相机拍摄图片时&#xff0c;边缘通常没有锯齿&#xff0c;因为边缘像素是一些前景和一些背景的混合。请考虑&#xff0c;…

5. 链接和加载(linker and loader)

链接和加载(linker and loader)&#xff1a; linker即链接器&#xff0c;它负责将多个.c编译生成的.o文件&#xff0c;链接成一个可执行文件或者是库文件&#xff1b; loader即加载器&#xff0c;它原本的功能很单一只是将可执行文件的段拷贝到编译确定的内存地址即可&#x…

英福康INFICON残余气体RGA General Chinese中文培训PPT课件

英福康INFICON残余气体RGA General Chinese中文培训PPT课件

【树上倍增】【割点】 【换根法】3067. 在带权树网络中统计可连接服务器对数目

作者推荐 视频算法专题 本文涉及知识点 树上倍增 树 图论 并集查找 换根法 深度优先 割点 LeetCode3067. 在带权树网络中统计可连接服务器对数目 给你一棵无根带权树&#xff0c;树中总共有 n 个节点&#xff0c;分别表示 n 个服务器&#xff0c;服务器从 0 到 n - 1 编号…

Java | 在消息对话框中显示文本

首先需要导入JOptionPane类&#xff0c;JOptionPane类属于Swing组件中的一种&#xff0c;其导入方式如下&#xff1a; import javax.swing.JOptionPane;可以使用JOptionPane的showMessageDialog方法显示消息文本。 参数格式&#xff1a; JOptionPane.showMessageDialog(paren…

【C语言】指针详细解读2

1.const 修饰指针 1.1 const修饰变量 变量是可以修改的&#xff0c;如果把变量的地址交给⼀个指针变量&#xff0c;通过指针变量的也可以修改这个变量。 但是如果我们希望⼀个变量加上⼀些限制&#xff0c;不能被修改&#xff0c;怎么做呢&#xff1f;这就是const的作⽤。 …

RK3568平台开发系列讲解(基础篇)注册字符设备

🚀返回专栏总目录 文章目录 一、字符设备初始化二、字符设备的注册和注销三、实验代码沉淀、分享、成长,让自己和他人都能有所收获!😄 注册字符设备可以分为两个步骤: 字符设备初始化字符设备的添加一、字符设备初始化 字符设备初始化所用到的函数为 cdev_init(…),在对…

解决QMYSQL driver not loaded问题

前言 之前都是在Qt5.51上开发&#xff0c;连接mysql数据库一直没有问题&#xff0c;换到5.15.2后一直报错 一查才发现\5.15.2\msvc2019_64\plugins\sqldrivers目录下没有qsqlmysql了&#xff0c;5.5.1是有的&#xff0c;5.15.2是要自己编译的。。。 下载源码 安装qt的时候没…