洛谷 P1966 火柴排队 题解
归并排序
很玄学的一道题目,用另类的方法求出逆序对的数量就可以AC
我的思路是这样的:
按照题目,输入数据用两个数组a,b储存,
同时,用另外两个数组c,d分别对应前面两个a,b储存,
就是前面两个的复制
然后在将复制出来的拍一下序,
用另一个类似桶的分别记录在a,b两个数组的数字里面,
某一个数是里面第几大的数
然后按照b里面的顺序,也就是里面的从第一个到最后一个的数在桶里面对应的
是第几个
给a标一下第一第二
比如b中的顺序是 :1 , 3 , 2 , 4
在a中的顺序是 :1 , 4 , 2 , 3
那么根据b的来看,1在b中是第一个所以a中的1对应的还是1
3在b中是第二个,所以a中3对应的是2
.....
这样弄完之后
a成了1 , 4 , 3 , 2
然后归并排序
这里为什么可以用归并排序呢?
归并排序板子是按照大小来排序的,这里却是将a按照b来排序的
所以b就可以看作是大小,是将a在b中的位置看做是大小,这样归并排序按照改
完后的大小排一下
同时记录逆序对,也就是每一个需要挪动的次数
记录和,这个和就是需要挪动的次数的总和
完整代码
#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
const int Max = 100005;
const int mo = 99999997;//注意必须要mo一旦忽视就会错掉两个点
int a[Max],b[Max];
int c[Max],d[Max];
int num1[Max],num2[Max];
int js[Max];
long long ans = 0;
int aa[Max];
void gui(int x,int y)//归并排序板子
{
if(x == y)return;
int mid = (x + y) >> 1;
gui(x,mid);gui(mid + 1,y);
int i = x,j = mid + 1;
int k = x;
while(i <= mid && j <= y)
{
if(a[i] <= a[j])aa[k ++] = a[i ++];
else aa[k ++] = a[j ++],ans += mid - i + 1,ans %= mo;;
}
while(i <= mid)
aa[k ++] = a[i ++];
while(j <= y)
aa[k ++] = a[j ++];
for(int i = x;i <= y;++ i)
a[i] = aa[i];
}
int main()
{
int n;
scanf("%d",&n);
for(int i = 1;i <= n;++ i)scanf("%d",&a[i]),c[i] = a[i];
for(int i = 1;i <= n;++ i)scanf("%d",&b[i]),d[i] = b[i];//输入数据
sort(d + 1,d + 1 + n);
for(int i = 1;i <= n;++ i)
num1[d[i]] = i;
sort(c + 1,c + 1 + n);
for(int i = 1;i <= n;++ i)
num2[c[i]] = i;
for(int i = 1;i <= n;++ i)
js[num1[b[i]]] = i;
for(int i = 1;i <= n;++ i)
a[i] = js[num2[a[i]]];//前面这一大坨都是预处理下标
gui(1,n);
cout << ans % mo <<endl;
return 0;
}
洛谷 P1966 火柴排队 题解的更多相关文章
- [NOIP2013提高&洛谷P1966]火柴排队 题解(树状数组求逆序对)
[NOIP2013提高&洛谷P1966]火柴排队 Description 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相 ...
- 洛谷p1966火柴排队题解
ps:鉴于你们的蒟蒻yxj实在太蒻辽, 所以, 看不懂也是正常的........ 树状数组 xxy学姐给我们讲的树状数组, 她讲的真的是太好啦!qwq!吹爆xxy 然后, 为了巩固自己, 硬着头皮写题 ...
- 【刷题】洛谷 P1966 火柴排队
题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2 其中 ai 表示 ...
- 洛谷 P1966 火柴排队 解题报告
P1966 火柴排队 题目描述 涵涵有两盒火柴,每盒装有 \(n\) 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: \(\s ...
- 洛谷——P1966 火柴排队&&P1774 最接近神的人_NOI导刊2010提高(02)
P1966 火柴排队 这题贪心显然,即将两序列中第k大的数的位置保持一致,证明略: 树状数组求逆序对啦 浅谈树状数组求逆序对及离散化的几种方式及应用 方法:从前向后每次将数插入到bit(树状数组)中, ...
- [洛谷P1966] 火柴排队
题目链接: 火柴排队 题目分析: 感觉比较顺理成章地就能推出来?似乎是个一眼题 交换的话多半会往逆序对上面想,然后题目给那个式子就是拿来吓人的根本没有卵用 唯一的用处大概是告诉你考虑贪心一波,很显然有 ...
- 洛谷 P1966 火柴排队
题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为:∑(ai−bi)2 其中ai 表示 ...
- 洛谷 P1966 火柴排队 —— 思路
题目:https://www.luogu.org/problemnew/show/P1966 首先,一个排列相邻交换变成另一个排列的交换次数就是逆序对数: 随便画一画,感觉应该是排个序,大的对应大的, ...
- 洛谷——P1966 火柴排队
https://www.luogu.org/problem/show?pid=1966 题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列 ...
随机推荐
- Python字符串格式化方式之format
format方式是在Python3引入了一个新的字符串格式化的方法,并且随后支持了Python2.7.这个新的字符串格式化方法摆脱了%操作符并且使得字符串格式化的语法更规范了.现在时候通过调用字符串对 ...
- Spring Cloud Zuul路由规则动态更新
背景 Spring Cloud Zuul 作为微服务的网关,请求经过zuul路由到内部的各个service,由于存在着新增/修改/删除服务的路由规则的需求,zuul的路由规则的动态变更功能 提供了 ...
- chocolatey install curl and netcat
chocolatey install curl and netcat 软件仓库 https://chocolatey.org/packages choco install curl choco ins ...
- Java中关于Math的几个取整方法的区别
1.Math.ceil() 向上取整 System.out.println(Math.ceil(3.4)); //输出4 System.out.println(Math.ceil(3.7)); / ...
- BUAA OO 2019 第四单元作业总结
目录 第四单元总结 总 UML UML 类图 UML 时序图 UML 状态图 架构设计 第十三次作业 第十四次作业 课程总结 历次作业总结 架构设计 面向对象方法理解 测试方法理解与实践 改进建议 尽 ...
- JavaScript HTML DOM元素节点常用操作接口
在文档对象模型 (DOM) 中,每个节点都是一个对象.DOM 节点有三个重要的属性 : 1. nodeName : 节点的名称 2. nodeValue :节点的值 3. nodeType :节点的类 ...
- Air for ANE:一星期的调试笔记
来源:http://blog.csdn.net/hero82748274/article/details/8656674 第一次尝试ANE的东西,让我感觉到很折腾人.adobe 出的这个方案虽然可以解 ...
- python 操作excel实现替换特定内容
本文介绍使用python语言,借助openyxl库来实现操作excel(xlsx)文件,实现替换特定内容的需求. 目前实现了3个小功能: 1. 全字匹配替换(mode1):(如:全字匹配 yocich ...
- Python学习日记(九) 装饰器函数
1.import time a.time.time() 获取到当前的时间,返回值为浮点型 import time print(time.time()) #1565422783.6497557 b.ti ...
- Linux之redis-sentinel
一,Redis-Sentinel介绍 Redis-Sentinel是redis官方推荐的高可用性解决方案,当用redis作master-slave的高可用时,如果master本身宕机,redis本身或 ...