NOIP2013提高组 T2 火柴排队
一开始看也想不到这居然要用到逆序对,归并排序。
先来看看题目:
涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度。 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2
其中 ai 表示第一列火柴中第 i 个火柴的高度,bi 表示第二列火柴中第 i 个火柴的高度。
每列火柴中相邻两根火柴的位置都可以交换,请你通过交换使得两列火柴之间的距离最小。请问得到这个最小的距离,最少需要交换多少次?如果这个数字太大,请输出这个最小交换次数对 99,999,997 取模的结果。
样例输入:
样例输出:
根据样例,设第二行的数为a{1,3,4,2},第三行的数为b{1,7,2,4}。对于题目要求通过交换使得两列火柴之间的的距离最小,满足这个要求只能使a中高度最小的火柴和b中高度最小的火柴搭配,第二小的和第二小的搭配。为了方便处理和后续的操作,把a和b进行离散化。离散化后,a{1,3,4,2},b{1,4,2,3}(离散化详解在*1)。遍历数组a,查找a[i]在数组b中所对应的位置(详细处理在*2)(需要用二分查找*3),存入新数组R,处理后为R{1,4,2,3},R[i]即为a[i]将要移到的位置,最小交换的次数正是R的逆序对数量。a中的3要移到R中3的位置,需要向右交换两次。相同的,可以这样认为,R中的3要向左移到a中的3的位置,则需要两次交换。R中3(即R[4])的逆序对数量为2,整个R的逆序对数量也为2。所求最少交换次数就是R的逆序对数量。使用归并排序。
时间复杂度(大概):O(7*(n log n)) 离散化:4*(n log n) ( 4次快排) 预处理二分查找+二分查找:2*(n log n) 归并排序:(n log n) 绝对不会超时qwq
*1:离散化前:a{1,3,4,2},b{1,7,2,4},给每个数组的每一个数标上序号,按照数组内容带上序号一起排序,重新编号,再根据之前标上的序号排回来恢复原状。
原始状态:
a | 1 | 3 | 4 | 2 |
序号 | 1 | 2 | 3 | 4 |
b | 1 | 7 | 2 | 4 |
序号 | 1 | 2 | 3 | 4 |
按照数组内容排序:
a | 1 | 2 | 3 | 4 |
序号 | 1 | 4 | 2 | 3 |
b | 1 | 2 | 4 | 7 |
序号 | 1 | 3 | 4 | 2 |
重新编号:
a | 1 | 2 | 3 | 4 |
序号 | 1 | 4 | 2 | 3 |
b | 1 | 2 | 3 | 4 |
序号 | 1 | 3 | 4 | 2 |
按照序号重新排序恢复:
a | 1 | 3 | 4 | 2 |
序号 | 1 | 2 | 3 | 4 |
b | 1 | 4 | 2 | 3 |
序号 | 1 | 2 | 3 | 4 |
完成,a离散化后和原来一样是因为样例特殊,参考b的离散化过程就好了。
*2:a[i]在数组b中所对应的位置。首先列出数组。
a | 1 | 3 | 4 | 2 |
b | 1 | 4 | 2 | 3 |
序号 | 1 | 2 | 3 | 4 |
R |
i=1的情况:a[i]为1,b中的1在b[1]中,b[1]对应的序号为1,所以R[i]为1,i=1,所以R[1]填上1。
i=2的情况:a[i]=3,b[4]=3,所以b中的3在b[4],对应序号为4。R[2]填上4
i=3:a[i]=4,b[2]=4,所以R[i]=2,R[3]填上2
i=4:a[i]=2,b[3]=2,所以R[4]=3
最终得R为:
R | 1 | 4 | 2 | 3 |
*3:如果查找a[i]在数组b中所对应的位置使用两重循环,则查找的时间复杂度为O(n^2),数据范围为n<=100000,光是查找就会超时,所以需要用二分排序。
贴上代码详细参考:
type
arr=array[..] of longint;
var
a1,b1,a,r:arr;
n,i,j:longint;
left,right,mid:longint;
ans:qword;
procedure qsort(var a,b:arr); //快排a数组,捆绑b数组
procedure sort(l,r: longint);
var
i,j,x,y: longint;
begin
i:=l;
j:=r;
x:=a[(l+r) div ];
repeat
while a[i]<x do
inc(i);
while x<a[j] do
dec(j);
if not(i>j) then
begin
y:=a[i];
a[i]:=a[j];
a[j]:=y;
y:=b[i];
b[i]:=b[j];
b[j]:=y;
inc(i);
dec(j);
end;
until i>j;
if l<j then
sort(l,j);
if i<r then
sort(i,r);
end;
begin
sort(,n);
end;
procedure mergesort(s,t:longint); //归并排序用来统计逆序对
var
mid,i,j,k:longint;
begin
if s=t then exit;
mid:=(s+t) div ;
mergesort(s,mid);
mergesort(mid+,t);
i:=s;
j:=mid+;
k:=s;
while (i<=mid) and (j<=t) do
if a[i]<=a[j] then
begin
r[k]:=a[i];
inc(i);
inc(k);
end
else
begin
r[k]:=a[j];
inc(j);
inc(k);
ans:=ans+(mid-i+); //逆序对
end;
while i<=mid do
begin
r[k]:=a[i];
inc(i);
inc(k);
end;
while j<=t do
begin
r[k]:=a[j];
inc(j);
inc(k);
end;
for i:=s to t do
a[i]:=r[i];
end;
begin
assign(input,'match.in');
assign(output,'match.out');
reset(input);
rewrite(output);
readln(n);
for i:= to n do
begin
read(a1[i]); //读入a数组
r[i]:=i; //编号
end;
qsort(a1,r); //按照数组内容排序
for i:= to n do a1[i]:=i; //重新编号 {离散化}
qsort(r,a1); //按照编号排序复原
readln;
for i:= to n do
begin
read(b1[i]); //读入b数组
r[i]:=i; //编号
end;
qsort(b1,r); //按照数组内容排序
for i:= to n do b1[i]:=i; //重新编号 {离散化}
qsort(r,b1); //按照编号排序复原 {for i:=1 to n do //处理a[i]在数组b中所对应的位置,使用两重循环,n到达1000时就会超时,所以舍弃
for j:=1 to n do
if b1[i]=a1[j] then
begin
a[i]:=j;
break;
end; } for i:= to n do r[i]:=i;
qsort(a1,r); //按照内容排序
for i:= to n do //遍历每个b[i]查找在a中的位置(这里a和b交换了,所以是查找b[i]在a中的位置)
begin
left:=;
right:=n;
mid:=(left+right) div ;
while (a1[mid]<>b1[i]) and (left<=right) do //二分查找,时间复杂度O(n log n)
begin
if a1[mid]>b1[i] then right:=mid-
else left:=mid+;
mid:=(left+right) div ;
end;
a[i]:=r[mid]; //此处的a数组即为上面所讲的R数组
end;
ans:=;
fillchar(r,sizeof(r),);
mergesort(,n); //归并排序统计逆序对
writeln(ans mod ); //依照题意对99999997取模
close(input);
close(output);
end.
NOIP2013提高组 T2 火柴排队的更多相关文章
- 洛谷P1966 [NOIP2013提高组Day1T2]火柴排队
P1966 火柴排队 题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi) ...
- 【NOIP2013提高组】火柴排队
https://www.luogu.org/problem/show?pid=1966 Σ(ai-bi)2=Σai2+Σbi2-2Σai*bi,要使Σ(ai-bi)2最小,则需2Σai*bi最大. 由 ...
- 【NOIP】提高组2013 火柴排队
[题意]两列n个火柴,分别有高度ai和bi(同一列高度互不相同),每次可以交换一列中的两个相邻火柴,定义距离为∑(ai-bi)^2,求使距离最小的最少交换次数,n<=10^5. [算法]逆序对 ...
- NOIP2013 提高组 Day1
https://www.luogu.org/problem/lists?name=&orderitem=pid&tag=83%7C30 期望得分:100+100+100=300 实际得 ...
- NOIP2013普及组 T2 表达式求值
OJ地址:洛谷P1981 CODEVS 3292 正常写法是用栈 #include<iostream> #include<algorithm> #include<cmat ...
- [NOIp2013提高组]积木大赛/[NOIp2018提高组]铺设道路
[NOIp2013提高组]积木大赛/[NOIp2018提高组]铺设道路 题目大意: 对于长度为\(n(n\le10^5)\)的非负数列\(A\),每次可以选取一个区间\(-1\).问将数列清零至少需要 ...
- noip2013 提高组
T1 转圈游戏 题目传送门 果不其然 第一题还是模拟题 一波快速幂解决问题 #include<cstdio> #include<cstring> #include<alg ...
- JZOJ2020年8月11日提高组T2 宝石
JZOJ2020年8月11日提高组T2 宝石 题目 Description 见上帝动了恻隐之心,天后也想显示一下慈悲之怀,随即从口袋中取出一块魔术方巾,让身边的美神维纳斯拿到后堂的屏风上去试试,屏风是 ...
- 【GDKOI2014】JZOJ2020年8月13日提高组T2 石油储备计划
[GDKOI2014]JZOJ2020年8月13日提高组T2 石油储备计划 题目 Description Input Output 对于每组数据,输出一个整数,表示达到"平衡"状态 ...
随机推荐
- python之描述符
描述符是将某种特殊类型的类实例指派给另一个类的属性,某种特殊类型的类就是这个类里面封装了get,set,delete这三个方法,可以将这个类指派给另一个类的某一个属性,这样就可以通过这三个方法对该属性 ...
- 【汇编语言】DOXBox 0.74 常用debug命令
1.查看.修改寄存器(r命令) ①-r ②-r ax(要修改的寄存器) -:m(输入想要改成什么值) 2.查看内存单元(d命令) ①-d 查看128个内存单元内容. ②-d 段地址:偏移地址 查看指 ...
- java中import详解
前言 import与package机制相关,这里先从package入手,再讲述import以及static import的作用. package package名称就像是我们的姓,而class名称就像 ...
- jQuery新版本没有了Toggle事件,两个按钮分别控制隐藏显示,同时这两个按钮点击也要互斥。
十二月没来得及整理发布,一直在草稿箱.现在已经2019年1月了... 需求大概是这样的 //XX点击事件 var flagBar = 0; $("#doNotBaseRate"). ...
- @Html.LabelFor 如何直接添加CSS样式
样式用的是bootstrap. 我想单独调整一下其中一个控件的样式,大概这个造型. @Html.LabelFor(m => m.DerivationRate, new { @class = &q ...
- fiddler抓包——IOS
fiddler抓包——IOS1.在fiddler上安装证书CertMakerhttps://www.telerik.com/fiddler/add-ons安装可以参考:https://blog.csd ...
- [Linux]出错处理errno
概述 公共头文件<errno.h>定义了一个整型值errno以及可以赋予它的各种常量. 大部分函数出错后返回-1,并且自动给errno赋予当前发生的错误枚举值. 需要注意的一点是,errn ...
- python正则表达式--flag修饰符、match对象属性
正则表达式—修饰符 正则表达式可以包含一些标志修饰符来控制匹配模式,用在正则表达式处理函数中的flag参数中,为可选参数. (1) re.I 全写(re.IGNORECASE) 表示使匹配时,忽略大小 ...
- 【JS】VUE学习
VUE的全家桶:vue-cli,vue-router,vue-resource,vuex 环境搭建:https://www.jianshu.com/p/32beaca25c0d 先码在这儿吧. htt ...
- JS & JQuery 动态处理select option
原文 出处http://www.51xuediannao.com/html+css/htmlcssjq/cssbuhuanhang.html 今天你问了我一个关于在<select>里动态添 ...