【BZOJ4295】[PA2015]Hazard

Description

有n个人在轮流玩赌博机,一开始编号为i的人有a[i]元钱。赌博机可以抽象为一个长度为m的仅包含1和-1的序列,若抽到1,那么你将得到1块钱;若抽到-1,你将输掉1块钱。
第1局,第1个人会抽到序列中的第1项;第2局,第2个人会抽到序列中的第2项;第3局,第3个人会抽到序列中的第3项......即:第i个人抽完后轮到第i+1个人去抽,特别地,第n个人抽完后轮到第1个人去抽。序列第i项被抽到之后,下一个被抽到的将会是第i+1项,特别地,序列第m项被抽到之后,下一个被抽到的将会是第1项。
如果在某一轮,有个人输光了所有的钱,那么这场赌博游戏就会结束,请求出游戏在哪一轮结束,或者判断这个游戏会永远进行下去。

Input

第一行包含一个正整数n(1<=n<=1000000),表示玩家的个数。
第二行包含n个正整数a[1],a[2],...,a[n](1<=a[i]<=1000000),依次表示每个玩家一开始持有的钱数。
第一行包含一个正整数m(1<=m<=1000000),表示序列的长度。
第四行包含一个长度为m的仅包含W和P的字符串,表示这个序列,其中W表示1,P表示-1。

Output

若游戏会永远进行下去,输出-1。否则输出游戏在哪一轮结束。

Sample Input

4
2 3 2 1
3
WPP

Sample Output

12

题解:显然我们应该找到循环节,然后将一个循环内的所有人放到一起考虑。假如我们已经拿出了某个循环中的所有人,我们先对于每个人,处理处他在转一圈时,收益的最小值(亏损的最大值);再处理出整个循环的权值和。那么如果权值和是负数,那么这个人肯定是先转若干圈,直到他的剩余钱数不足这个亏损的最大值,然后找到他第一个把钱输光的位置即可。如果权值和是正数,我们看这个人的钱数是否大于这个亏损的最大值。如果大于,则这个人永远不会输光,否则找到他第一个把钱输光的位置。

以上内容都可以通过前缀和,前缀最大值,前缀pre,后缀和,后缀最大值,后缀nxt搞定。特别地,如果n>m或m>n都要特殊考虑一下。

#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
const int maxn=1000010;
typedef long long ll;
int n,m;
ll ans;
int A[maxn],B[maxn],C[maxn],v[maxn],vis[maxn],s[maxn],s1[maxn],s2[maxn],m1[maxn],m2[maxn],p[maxn];
int memp[maxn<<1],*last=memp+maxn,mn[maxn];
char str[maxn];
inline int abs(int x) {return x>0?x:-x;}
void work(int S)
{
int i,j,L=0,sum=0;
for(i=S;!vis[i];i=(i+n)%m,L++) p[L]=i,v[L]=(str[i]=='W')?1:-1,sum+=v[L],vis[i]=1;
s2[L]=m2[L]=0;
for(i=0;i<L;i++) s1[i]=s1[i-1]+v[i],m1[i]=min(m1[i-1],s1[i]);
for(i=L-1;i>=0;i--) s2[i]=s2[i+1]+v[i],m2[i]=max(m2[i+1],s2[i]),mn[i]=min(s2[i]+m1[i-1],s2[i]-m2[i+1]);
if(sum<0)
{
sum=-sum;
for(i=0;i<L;i++)
{
j=p[i];
if(A[j]!=-1&&A[j]+mn[i]>0) B[j]=(A[j]+mn[i]+sum-1)/sum,A[j]-=B[j]*sum;
}
}
for(i=0;i<=L;i++) last[i]=last[-i]=-2;
for(i=0;i<L;i++)
{
j=p[i];
if(last[s1[i-1]]==-2) last[s1[i-1]]=i-1;
if(A[j]!=-1&&A[j]+mn[i]<=0&&last[-s2[i]-A[j]]!=-2)
ans=min(ans,(last[-s2[i]-A[j]]+L-i+(ll)B[j]*L)*n+C[j]);
}
for(i=0;i<=L;i++) last[i]=last[-i]=-2;
for(i=L-1;i>=0;i--)
{
j=p[i],last[s2[i+1]]=i+1;
if(A[j]!=-1&&A[j]+mn[i]<=0&&last[s2[i]+A[j]]!=-2)
ans=min(ans,(last[s2[i]+A[j]]-i-1+(ll)B[j]*L)*n+C[j]);
}
}
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-') f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
return ret*f;
}
int main()
{
//freopen("bz4295.in","r",stdin);
n=rd();
int i;
for(i=0;i<n;i++) A[i]=rd();
m=rd();
scanf("%s",str);
for(i=0;i<m;i++) C[i]=i+1;
for(i=m;i<n;i++) if(A[i]<A[i%m]) A[i%m]=A[i],C[i%m]=i+1;
for(i=n;i<m;i++) A[i]=-1;
ans=1ll<<60;
for(i=0;i<m;i++) if(!vis[i]) work(i);
if(ans==(1ll<<60)) printf("-1");
else printf("%lld",ans);
return 0;
}

【BZOJ4295】[PA2015]Hazard 乱搞的更多相关文章

  1. BZOJ4295 : [PA2015]Hazard

    第i轮,a[i%n]+=b[i%m]. 枚举i,计算它变为0的次数,假设为t,那么有t=i+kn. 对于所有的i和k,(i+kn)%m形成了若干个总长度为m的环. 对于每个a[i],先在环中求出一轮最 ...

  2. URAL 1827 Indigenous Wars(排序、乱搞)

    题意:给一个长度为n数组{a[i]}.有m个操作Ti,Si,Li表示找以Ti值结束,以Si值开始,长度为Li的连续子串.找到后,将区间的答案值设为1.一开始答案值全部为0.最后输出n个答案值. 好久没 ...

  3. UVA 11853 [dfs乱搞]

    /* 大连热身E题 不要低头,不要放弃,不要气馁,不要慌张 题意: 在1000×1000的格子内有很多个炮弹中心,半径给定. 为某人能否从西部边界出发,从东部边界走出. 不能输出不能,能的话输出最北边 ...

  4. Codeforces 732e [贪心][stl乱搞]

    /* 不要低头,不要放弃,不要气馁,不要慌张 题意: 给n个插座,m个电脑.每个插座都有一个电压,每个电脑都有需求电压. 每个插座可以接若干变压器,每个变压器可以使得电压变为x/2上取整. 有无限个变 ...

  5. 【BZOJ-4692】Beautiful Spacing 二分答案 + 乱搞(DP?)

    4692: Beautiful Spacing Time Limit: 15 Sec  Memory Limit: 128 MBSubmit: 46  Solved: 21[Submit][Statu ...

  6. 【BZOJ-3578】GTY的人类基因组计划2 set + map + Hash 乱搞

    3578: GTY的人类基因组计划2 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 367  Solved: 159[Submit][Status][ ...

  7. 【BZOJ-2937】建造酿酒厂 前缀和 + 展环为链 + 乱搞

    2937: [Poi2000]建造酿酒厂 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 70  Solved: 24[Submit][Status][D ...

  8. SCOI 2013 密码 & 乱搞

    题意: Fish 是一条生活在海里的鱼.有一天他很无聊,就到处去寻宝.他找到了位于海底深处的宫殿,但是一扇带有密码锁的大门却阻止了他的前进.通过翻阅古籍,Fish 得知了这个密码的相关信息:1. 该密 ...

  9. 种树 & 乱搞

    题意: 在一个(n+1)*(m+1)的网格点上种k棵树,树必须成一条直线,相邻两棵树距离不少于D,求方案数. SOL: 这题吧...巨坑无比,本来我的思路是枚举每一个从(0,0)到(i,j)的矩形,然 ...

随机推荐

  1. 用Visual studio2012在Windows8上开发内核驱动监视线程创建

    在Windows NT中,80386保护模式的“保护”比Windows 95中更坚固,这个“镀金的笼子”更加结实,更加难以打破.在Windows 95中,至少应用程序I/O操作是不受限制的,而在Win ...

  2. tomcat的web项目的远程热发布

    已经发布的项目war包更改了怎么办?我常用的做法是: .把war包上传服务器 .远程登录服务器后台 .停止tomcat服务 .用新的war包替换老的war包 .启动tomcat服务 后来才知道原来to ...

  3. POJ 3279 Fliptile (二进制+搜索)

    [题目链接]click here~~ [题目大意]: 农夫约翰知道聪明的牛产奶多. 于是为了提高牛的智商他准备了例如以下游戏. 有一个M×N 的格子,每一个格子能够翻转正反面,它们一面是黑色,还有一面 ...

  4. 统计重复字符串 如 eeefffkkkhjk 得到如下结果 3e3f3khjk;

    //统计重复字符串 如 eeefffkkkhjk 得到如下结果 3e3f3khjk; string zipStr = Console.ReadLine(); var charList = zipStr ...

  5. Mysql 比较运算符详解

    熟悉了最简单的算术运算符,再来看一下比较运算符.当使用SELECT 语句进行查询时,MySQL允许用户对表达式的左边操作数和右边操作数进行比较,比较结果为真,则返回1,为假则返回0,比较结果不确定则返 ...

  6. AngularJS 中 Provider 的用法及区别

    在一个分层良好的 Angular 应用中,Controller 这一层应该很薄.也就是说,应用里大部分的业务逻辑和持久化数据都应该放在 Service 里. 为此,理解 AngularJS 中的几个 ...

  7. Atitit.分布式远程调用  rpc  rmi  CORBA的关系

    Atitit.分布式远程调用  rpc  rmi  CORBA的关系 1. 远程调用(包括rpc,rmi,rest)1 2. 分布式调用大体上就分为两类,RPC式的,REST式的1 3. RPC(远程 ...

  8. Xcode常见问题

    今天真机测试的时候,突然出现了这个错误:  not have an architecture that “Administrator”的 iPhone (3) can execute. 原因是我刚刚修 ...

  9. 跟着百度学PHP[8]-setcookie的其他参数学习

    setcookie(name,value,time,path,domain,secure) 第四个参数是cookie的有效路径.顾名思义就是在那个路径下是有效的.当然了其子目录下也是有效的.比如 /h ...

  10. [systemd]Linux系统启动之systemd

    参照:https://wiki.debian.org/systemd 最近在添加板子应用程序自启动的时候,发现在rcN.d中的符号链接并没有用,文件系统为Debian Jessie 8, 后来从同事那 ...