[Educational Round 10][Codeforces 652F. Ants on a Circle]
题目大意:\(n\)个蚂蚁在一个大小为\(m\)的圆上,每个蚂蚁有他的初始位置及初始面向,每个单位时间蚂蚁会朝着当前面向移动一个单位长度,在遇到其它蚂蚁时会立刻掉头。求经过\(t\)个单位时间后每一个蚂蚁的所在位置
题解:首先可以发现,最终答案其实是在不考虑碰撞下得出答案的一个排列,而且蚂蚁们的相对位置是不会改变的。所以如果求出了其中任意一个蚂蚁的位置,就能求出最终的答案。
为了方便起见,先默认所有蚂蚁的位置是按升序排的,且第一个蚂蚁的位置为\(0\)
对于碰到同伴就掉头这个情况,可以换一种视角来考虑,即碰到同伴时,不掉头,而是交换相遇两蚂蚁的编号,这样子做从结果上看二者并没有差别
对于所有蚂蚁,就以第一个蚂蚁为例,假设其初始面向是向右,则当他遇到其他蚂蚁时,会和他右边的蚂蚁交换编号。又由蚂蚁们的相对位置不变,可以得出他右边蚂蚁的编号始终是比他自己大\(1\)的,因此可以得出,每次碰撞带来的后果就是使其编号加一。又因为在走过一圈后,所有的蚂蚁都回到了初始的位置,且面向未发生改变,所以每一圈编号的变化量是固定的。所以可以得出在经过\(t\)时间后,该蚂蚁的编号发生了多少改变,并与该蚂蚁的最终位置相匹配即可。
这种做法存在一种问题就是,若这个蚂蚁的最终位置与其他蚂蚁的位置重复,则无法确定他具体在哪个编号上。这时可以找出那个和该蚂蚁终点相同的蚂蚁,再做一遍相同的操作,比较下两个蚂蚁的编号就好了。
虽然代码看上去很长,但实际上\(f**k()\)里的内容都是照抄下面的东西,写起来还是挺简单的
#include<bits/stdc++.h>
using namespace std;
#define N 600001
#define LL long long
#define mp make_pair
LL n,m,t,d,x,mn,cnt,b[N],r[N],r2[N],f[N],ans[N];
pair<LL,LL>a[N];
bool cmp(LL x,LL y){return a[x]<a[y];}
bool cmp2(LL x,LL y){return f[x]<f[y];}
LL get()
{
char ch=getchar();
while(ch!='L' && ch!='R')
ch=getchar();
return ch=='L'?-:;
}
void fuck()
{
for(LL i=;i<n;i++)
f[i]=(a[i].first+(t%m)*(m+a[i].second))%m;
LL X=;
for(LL i=;i<n;i++)
if(f[i]==f[])X=i;
LL c=a[X].first;
for(LL i=;i<n;i++)
a[i].first=(a[i].first+m-c)%m;
sort(a,a+n);
cnt=;
for(LL i=;i<n;i++)
if(a[i].second!=a[].second)
{
b[cnt++]=a[i].first;
b[cnt++]=a[i].first+m;
}
sort(b,b+cnt);
for(LL i=;i<n;i++)
f[i]=(a[i].first+(t%m)*(m+a[i].second))%m;
LL dd=lower_bound(b,b+cnt,f[]*)-b;
if(b[dd]==f[]* && a[].second>)dd++;
if(a[].second< && f[])dd=cnt-dd;
dd+=cnt*((t/m)%n),dd%=n;
if(a[].second<)dd=(n-dd)%n;
for(LL i=;i<n;i++)
a[i].first=(a[i].first+c)%m;
for(LL i=;i<n;i++)
f[i]=(a[i].first+(t%m)*(m+a[i].second))%m;
sort(f,f+n);
if((d+)%n==(dd+X)%n && f[x]==f[(x+n-)%n])x=(x+n-)%n;
if(d==(dd+X+)%n && f[x]==f[(x+)%n])x=(x+)%n;
}
int main()
{
scanf("%I64d%I64d%I64d",&n,&m,&t);
for(LL i=;i<n;i++)
{
r[i]=r2[i]=i;
scanf("%I64d",&a[i].first);
a[i].second=get();
}
sort(r,r+n,cmp);
sort(a,a+n);
mn=a[].first;
for(LL i=;i<n;i++)
a[i].first-=mn;
for(LL i=;i<n;i++)
if(a[i].second!=a[].second)
{
b[cnt++]=a[i].first;
b[cnt++]=a[i].first+m;
}
sort(b,b+cnt);
for(LL i=;i<n;i++)
f[i]=(a[i].first+(t%m)*(m+a[i].second))%m;
d=lower_bound(b,b+cnt,f[]*)-b;
if(b[d]==f[]* && a[].second>)d++;
if(a[].second< && f[])d=cnt-d;
d+=cnt*((t/m)%n),d%=n;
if(a[].second<)d=(n-d)%n;
sort(r2,r2+n,cmp2);
sort(f,f+n);
for(LL i=;i<n;i++)
if(r2[i]==)x=i;
if(f[x]==f[(x+)%n] || f[x]==f[(x+n-)%n])fuck();
for(LL i=;i<n;i++)
ans[r[(d+i)%n]]=f[(x+i)%n];
for(LL i=;i<n;i++)
printf("%I64d%c",(ans[i]+mn-)%m+,i<n-?' ':'\n');
return ;
}
[Educational Round 10][Codeforces 652F. Ants on a Circle]的更多相关文章
- Codeforces 652F Ants on a Circle
Ants on a Circle 感觉这个思路好巧妙啊. 我们能发现不管怎么碰撞,初始态和最终态蚂蚁间的相对顺序都是一样的, 并且所占的格子也是一样的, 那么我们就只需要 找到其中一个蚂蚁的最终位置就 ...
- [Educational Round 5][Codeforces 616F. Expensive Strings]
这题调得我心疲力竭...Educational Round 5就过一段时间再发了_(:з」∠)_ 先后找了三份AC代码对拍,结果有两份都会在某些数据上出点问题...这场的数据有点水啊_(:з」∠)_[ ...
- [Educational Round 3][Codeforces 609E. Minimum spanning tree for each edge]
这题本来是想放在educational round 3的题解里的,但觉得很有意思就单独拿出来写了 题目链接:609E - Minimum spanning tree for each edge 题目大 ...
- [Educational Round 3][Codeforces 609F. Frogs and mosquitoes]
这题拖了快一周_(:з」∠)_就把这货单独拿出来溜溜吧~ 本文归属:Educational Codeforces Round 3 题目链接:609F - Frogs and mosquitoes 题目 ...
- [Educational Round 17][Codeforces 762F. Tree nesting]
题目连接:678F - Lena and Queries 题目大意:给出两个树\(S,T\),问\(S\)中有多少连通子图与\(T\)同构.\(|S|\leq 1000,|T|\leq 12\) 题解 ...
- [Educational Round 13][Codeforces 678F. Lena and Queries]
题目连接:678F - Lena and Queries 题目大意:要求对一个点集实现二维点对的插入,删除,以及询问\(q\):求\(max(x\cdot q+y)\) 题解:对每个点集内的点\(P( ...
- [Educational Round 59][Codeforces 1107G. Vasya and Maximum Profit]
咸鱼了好久...出来冒个泡_(:з」∠)_ 题目连接:1107G - Vasya and Maximum Profit 题目大意:给出\(n,a\)以及长度为\(n\)的数组\(c_i\)和长度为\( ...
- CF - 652F Ants on a Circle
题目传送门 题解: 先观察蚂蚁相撞, 可以发现, 如果我们将相撞的2个蚂蚁互换位置的话,蚂蚁相当于没有碰撞体积,直接穿过去了.所以我们可以直接计算出最终哪些位置上会有蚂蚁. 接下来就需要知道蚂蚁们的最 ...
- Educational Codeforces Round 10
A:Gabriel and Caterpillar 题意:蜗牛爬树问题:值得一提的是在第n天如果恰好在天黑时爬到END,则恰好整除,不用再+1: day = (End - Begin - day0)/ ...
随机推荐
- STM32F0使用LL库实现SHT70通讯
在本次项目中,限于空间要求我们选用了STM32F030F4作为控制芯片.这款MCU不但封装紧凑,而且自带的Flash空间也非常有限,所以我们选择了LL库实现.本篇我们将基于LL库采用模拟I2C接口的方 ...
- 动态的加载显示oracle警告日志文件内容
Last login: Fri Jan 25 00:37:47 2019 from oracle [root@oracle ~]# su - oracle [oracle@oracle ~]$ sql ...
- 《剑指offer》左旋转字符串
本题来自<剑指offer> 反转链表 题目: 思路: C++ Code: Python Code: 总结:
- C# .net 填充无效,无法被移除 微信小程序解密失败的解决办法
微信小程序获取用户信息诸如unionId的时候需要解密,如果遇到偶然的解密失败(填充无效,无法被移除),原因很有可能是session_key错误, 也是就你用作解密的session_key并不是微信用 ...
- react安装 项目构建
1.nodejs安装 下载安装包,解压.如果是已编译文件,在/etc/profile中设置PATH(/etc/profile文件中的变量设置,所有用户可用,但需求重启服务器),并source /etc ...
- pyqt5与QML开发小结
遇见的坑 qt 5.11 与 qt 5.12 中Qquick的差异还是蛮大的,由开发环境:Pyqt5.11 + Qt5.12 部署到 Pyqt5.11 + Qt5.11时遇到以下问题: 1.当一个It ...
- 自己动手造拖拉机轮子系列 -(react-loadable)
最新消息:react官方已支持懒加载https://reactjs.org/docs/code-splitting.html#reactlazy 文章webpack分片chunk加载原理中深入探究了异 ...
- c/c++再学习:排序算法了解
1.冒泡排序 冒泡排序是一种简单的排序算法.它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来.走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成. ...
- 不用框架,原生使用python做注册接口/登陆接口/充值接口的测试,做的数据/代码分离
注意充值时候,cookie的处理方法,还是原来的三种方法 1.操作数据文件 do_exclel.py # -*- conding:utr-8 -*- #@Time :2018/11/8 22:46 ...
- 欧朋Opera 浏览器(打不开百度)提示“您的连接不是私密连接”,解决办法
它网页报错SSL.提示“您的连接不是私密连接” 打开opera://net-internals/#hsts,操作如下图片,三步以后,ok 如果是其他外国浏览器,用这个办法也有效,可以把前面的 oper ...