@codeforces - 1214F@ Employment
@description@
有 m 个城市围成一个圆环,编号为 1~m。
某公司有 n 个职员住在 m 个城市,依次住在编号为 a1, a2, ..., an 的城市(可能住在同一城市)。且该公司有 n 个工作场所,依次在 b1, b2, ..., bn(一样可以在同一城市)。
现在要将职员与城市一一匹配,使得每个职员的居所与工作场所距离之和最小,并输出方案。
Input
第一行两个整数 m, n(1 <= m <= 10^9, 1 <= n <= 200000),表示圆环上的城市个数以及职员个数。
第二行 n 个整数 a1, a2, ..., an(1 <= ai <= m),表示职员居住的城市编号
第三行 n 个整数 b1, b2, ..., bn(1 <= bi <= m),表示工作场所所在的城市编号。
Output
第一行输出最小距离总和。
第二行输出匹配方案,第 i 个数表示第 i 个职员工作的场所编号。
Examples
Input
10 3
1 5 5
10 4 6
Output
3
1 2 3
Input
10 3
1 4 8
8 3 6
Output
4
2 3 1
Note
第一个样例中,1 - 10, 5 - 4, 5 - 6 的距离都为 1。
第二个样例中,1 - 3, 4 - 6, 8 - 8 的距离分别为 2, 2, 0。
@solution@
巧妙地见过类似的原题。。。
然而我在考场上还是没有做出来,真是丢人。
先将 a, b 排好序。
首先将 bi 扩张成三倍,分别为 bi - m, bi, bi + m。则 ai 与 bj 之间在圆环上的距离肯定为 ai 到 bj - m, bj, bj + m 之中的某一个的数轴上的距离。
记这个扩张过后的 b 为 b‘。
假如一个方案中 ai 匹配 bx,aj 匹配 by,且 ai < aj, bx > by(即匹配出现交叉),则交换匹配变成 ai 匹配 by,aj 匹配 bx 会更优(分类讨论众多情况即可验证)。
同时还可以在原先的圆环上验证一个结论:如果 a1 匹配 b[x],则 a2 应依次匹配 b[x+1],a3 匹配 b[x+2],...。
贪心地得到了这两个结论后,我们最终有效的方案只有 2*n + 1 种。只要能有效地维护出每种方案的距离总和就可以解决这道题。
定义 ans[i] 表示 a1 匹配 i 的距离总和。
两个点 i, j 的距离定义为 |i - j|,其中 i >= j 时为 i - j,否则为 j - i。
一开始 a1 匹配 b1,匹配的点对之间距离总为 ai - bi。将 a1 对应的匹配往后移动时,ai - bj 就将变为 bj - ai。
于是对于每一个 ai,对一开始某个连续的区间的贡献为 +ai,对后面某个连续的区间的贡献为 -ai;bi 对一开始某个连续的区间的贡献 -bi,对后面某个连续的区间的贡献为 +bi。
找出那个变化的点(即 b[j] <= a[i] < b[j+1]),做一个差分即可。
@accepted code@
#include<cstdio>
#include<algorithm>
using namespace std;
typedef long long ll;
const int MAXN = 200000;
const ll INF = (1LL<<60);
struct node{
int pos; ll key;
friend bool operator < (node a, node b) {
return a.key < b.key;
}
}a[MAXN + 5], b[3*MAXN + 5];
ll sum[3*MAXN + 5];
int res[MAXN + 5];
int n; ll m;
int main() {
scanf("%lld%d", &m, &n);
for(int i=1;i<=n;i++)
scanf("%d", &a[i].key), a[i].pos = i;
sort(a + 1, a + n + 1);
a[n + 1].key = INF;
for(int i=1;i<=n;i++)
scanf("%d", &b[i].key), b[i].pos = i;
sort(b + 1, b + n + 1);
for(int i=1;i<=n;i++) {
b[2*n + i] = b[n + i] = b[i];
b[i].key -= m, b[2*n + i].key += m;
}
int p = 1;
for(int i=1;i<=n;i++) {
while( b[p].key <= a[i].key ) p++;
sum[0] += a[i].key;
if( p - i + 1 <= 2*n + 1 )
sum[p - i + 1] -= 2*a[i].key;
}
p = 1;
for(int i=1;i<=3*n;i++) {
while( a[p].key < b[i].key ) p++;
sum[max(0, i - n + 1)] -= b[i].key;
if( i - p + 2 <= 2*n + 1 )
sum[i - p + 2] += 2*b[i].key;
sum[min(i + 1, 2*n + 2)] -= b[i].key;
}
int ans = 1;
for(int i=1;i<=2*n+1;i++) {
sum[i] += sum[i-1];
if( sum[i] < sum[ans] )
ans = i;
}
for(int i=1;i<=n;i++)
res[a[i].pos] = b[ans+i-1].pos;
printf("%lld\n", sum[ans]);
for(int i=1;i<=n;i++)
printf("%d ", res[i]);
}
@details@
当时在网上随便找了一个题解,改改改写写写结果怎样都过不了。
后来自己一写就过了 =_=。
@codeforces - 1214F@ Employment的更多相关文章
- python爬虫学习(5) —— 扒一下codeforces题面
上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...
- 【Codeforces 738D】Sea Battle(贪心)
http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...
- 【Codeforces 738C】Road to Cinema
http://codeforces.com/contest/738/problem/C Vasya is currently at a car rental service, and he wants ...
- 【Codeforces 738A】Interview with Oleg
http://codeforces.com/contest/738/problem/A Polycarp has interviewed Oleg and has written the interv ...
- CodeForces - 662A Gambling Nim
http://codeforces.com/problemset/problem/662/A 题目大意: 给定n(n <= 500000)张卡片,每张卡片的两个面都写有数字,每个面都有0.5的概 ...
- CodeForces - 274B Zero Tree
http://codeforces.com/problemset/problem/274/B 题目大意: 给定你一颗树,每个点上有权值. 现在你每次取出这颗树的一颗子树(即点集和边集均是原图的子集的连 ...
- CodeForces - 261B Maxim and Restaurant
http://codeforces.com/problemset/problem/261/B 题目大意:给定n个数a1-an(n<=50,ai<=50),随机打乱后,记Si=a1+a2+a ...
- CodeForces - 696B Puzzles
http://codeforces.com/problemset/problem/696/B 题目大意: 这是一颗有n个点的树,你从根开始游走,每当你第一次到达一个点时,把这个点的权记为(你已经到过不 ...
- CodeForces - 148D Bag of mice
http://codeforces.com/problemset/problem/148/D 题目大意: 原来袋子里有w只白鼠和b只黑鼠 龙和王妃轮流从袋子里抓老鼠.谁先抓到白色老鼠谁就赢. 王妃每次 ...
随机推荐
- myql 配置项
提高数据插入速度方法 bulk_insert_buffer_size 默认:8M (8*1024*1024) 参考网址:https://stackoverflow.com/questions/2030 ...
- [转]js设计模式—发布订阅模式
发布—订阅模式又叫观察者模式,它定义对象间的一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都将得到通知.在javascript开发中,一般用事件模型来替代传统的发布—订阅模式.本文将 ...
- 使用Jedis操作Redis-使用Java语言在客户端操作---List类型
在Redis中,List类型是按照插入顺序排序的字符串链表.和数据结构中的普通链表一样,我们可以在其头部(left)和尾部(right)添加新的元素.在插入时,如果该键并不存在,Redis将为该键创建 ...
- 网站被攻击扫描SQL注入的日常记录
我发了个博客,泄露了域名之后,便有人疯狂的尝试攻击我的站点,奈何我防守做得比较好,直接把网段封了,看到403还锲而不舍,我真是想给他颁奖了 查看ua,发现很多sqlmap的ua,肯定会是被刷了,只是运 ...
- AndroidStudio离线打包MUI集成JPush极光推送并在java后端管理推送
1.AndroidStudio离线打包MUI 如何离线打包请参看上篇随笔<AndroidStudio离线打包MUI> 2.集成极光推送 官方文档:https://docs.jiguang. ...
- 微信公众号系统在Linux下的部署操作
ps -ef | grep tomcat 查看tomcat进程
- Inno Setup生成桌面快捷方式
在做项目的时候,需要打包成exe安装包.先前使用的是vs来打包,生成了setup.exe 和 *.msi的安装文件,不过也算顺利. 后因为要求采取 Inno Setup来打包程序,其中遇到个创建快捷方 ...
- KiCad 5.1.4 无法覆铜?
KiCad 5.1.4 无法覆铜? 群里有小伙伴发现焊盘无法覆铜,只能靠手工连接. 这就奇怪了,正常情况不会出现的这种现象的. 分析了很多可能,比较间隙太小,有试着调过,但还是连接不上. 把封装的所有 ...
- COGS-2638 区间与,异或,询问max
本篇题解参考了这个博客 题目链接 我们利用线段树来维护区间第最大值,考虑如何修改 每一次进行与操作时只有z的二进制为0的位会产生影响 每一次进行或操作时只有z的二进制为1的位会产生影响 所以只要该区间 ...
- C++继承与构造函数、复制控制
每个派生类对象由派生类中定义的(非static)成员加上一个或多个基类子对象构成,因此,当构造.复制.赋值和撤销派生类型对象时,也会构造.复制.赋值和撤销这些基类子对象. 构造函数和复制控制成员不能继 ...