codeforces733-C. Epidemic in Monstropolis 贪心加链表
题意
现在有一个怪兽序列a[i],权值大的怪兽可以吃权值小的怪兽,吃完之后权值大的怪兽的权值会变成两者权值的和,相邻的怪兽才能吃
吃完之后,位置合并,队列前移,从左到右重新编号,重复这一过程,
然后给你一个怪兽序列b[i],问你a[i]怎么操作能变成b[i],如果能操作,输出操作序列,如果不能操作就输出NO
难点一,划分序列,由于b[i]的概念类似于前缀和,一旦b[i]是确定的数,那么a[i]序列的划分就是唯一的,b[i]和a[i]的和相等
b[i]序列实际上就是对序列a[i]的唯一划分
难点二,如何判定由b[i]划分的小区间一定能全部合并?我们的办法是,找最容易满足的条件,如果你能养起来尽可能大的怪兽
,一旦它成为了最大值,那么这个小区间一定是可以合并的,这个贪心策略看起来是显然的(笑,所以我们每次尽量合成最大的怪兽
如果这样都合并不了整个区间,那么这个区间一定是不可合并的,我表示不会严格数学证明
难点三,如果用数组这个数据结构,我发现合并这个操作,我完全不知道怎么写,加上vis数组?哪个被合并就哪个记为被访问?
那么我们每次要找到两个没有被访问且连续的元素然后合并权值,操作vis,也不是不行,我觉得这个写法太tm恶心了
我选择用链表,权值用a[i]存,nxt[i]存储每个元素的下一跳,合并操作就变成了连跳操作,nxt[i]=nxt[nxt[i]],
但是对于数序号来说,我们要另外写一个函数来O(n),获取该元素的rank,还好n比较小
难点三、输出最终的答案序列
我用了vector,和结构体存储了位置了操作字符,不是很简单,有机会去看看大神们是怎么写的
难点四、考虑b[i]序列是否是合法序列,少考虑了两种,这个太致命了,一开始我是有想法的,后来忘了填坑了,以后觉得不放心的
地方都要加一个注释才行,具体坑点详见代码
难点五、注意合并操作的更新位置,要想记录当前操作,需要先记录再更新,更新后再记录就是错的了,不是我们想要的信息
#include <cstdio>
#include <vector>
#define ll long long
const int maxn=;
int n,k;
int a[maxn],b[maxn],flag[maxn];
int nxt[maxn];
struct node{
int pos;
char type;
//0 left 1 right
};//结构体后面忘了加分号
std::vector<node> ans;
void init(int left,int right){
register int i;
for(i=left;i<right;++i){
nxt[i]=i+;
}
nxt[right]=-;
}
bool check(int left,int right){
if(nxt[left]!=-) return false;
return true;
}
int findIndex(int left,int index){
int cnt=;
register int i;
for(i=left;i!=-;i=nxt[i]){
cnt++;
if(index==i){
return cnt;
}
}
return -;
//error
}
int solve(int left,int right,int pre){
ll mx=-;
int index,first=;
register int i;
for(i=left;i!=-;i=nxt[i]){
if(nxt[i]!=-&&a[i]!=a[nxt[i]]){
if(first){
first=;
mx=a[i]+a[nxt[i]];
index=i;
}
else{
ll temp=a[i]+a[nxt[i]];
if(temp>mx){
mx=temp;
index=i;
}
}
}
}
if(mx!=-){
int pos1=findIndex(left,index);
int pos2=findIndex(left,nxt[index]);
if(a[index]>a[nxt[index]]){
ans.push_back((node){pre+pos1,'R'});
}
else{
ans.push_back((node){pre+pos2,'L'});
}
a[index]+=a[nxt[index]];//这两句话位置不对,之前做的早了
nxt[index]=nxt[nxt[index]];
}
return mx;
}
int main(){
scanf("%d",&n);
register int i;
ll suma=,sumb=;
for(i=;i<n;++i){
scanf("%d",a+i);
suma+=a[i];
}
scanf("%d",&k);
for(i=;i<k;++i){
scanf("%d",b+i);
sumb+=b[i];
}
if(suma!=sumb) {
printf("NO\n");
return ;
}
ll temp=;
int cur=,cnt=;
for(i=;i<n;++i){
temp+=a[i];
if(temp==b[cur]){
flag[cnt++]=i;
++cur;
temp=;
}
else if(temp>b[cur]){
printf("NO\n");//NO 打成 No
return ;
}
}
if(cur!=k) {
printf("NO\n");
return ;
}
register int j;
int No=;
for(i=;i<cnt;++i){
int left,right=flag[i];
if(i==){
left=;
}
else{
left=flag[i-]+;
}
init(left,right);
while(!check(left,right)){
int p=solve(left,right,i);
if(p==-){
No=;
break;
}
}
if(No){
break;
}
}
if(No) printf("NO\n");
else{
printf("YES\n");
for(i=;i<ans.size();++i){
node t=ans[i];
printf("%d %c\n",t.pos,t.type);
}
}
return ;
}
codeforces733-C. Epidemic in Monstropolis 贪心加链表的更多相关文章
- CF733C Epidemic in Monstropolis[模拟 构造 贪心]
C. Epidemic in Monstropolis time limit per test 1 second memory limit per test 256 megabytes input s ...
- Epidemic in Monstropolis
Epidemic in Monstropolis 题目链接:http://codeforces.com/contest/733/problem/C 贪心 新序列的m个数肯定是由原序列的连续的m个子序列 ...
- 【16.52%】【codeforces 733C】Epidemic in Monstropolis
time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...
- Codeforces Round #378 (Div. 2) C. Epidemic in Monstropolis 模拟
C. Epidemic in Monstropolis time limit per test 1 second memory limit per test 256 megabytes input s ...
- Codeforces Round #378 (Div. 2)-C. Epidemic in Monstropolis
C. Epidemic in Monstropolis time limit per test 1 second memory limit per test 256 megabytes input s ...
- Codeforces 733C:Epidemic in Monstropolis(暴力贪心)
http://codeforces.com/problemset/problem/733/C 题意:给出一个序列的怪兽体积 ai,怪兽只能吃相邻的怪兽,并且只有体积严格大于相邻的怪兽才能吃,吃完之后, ...
- BZOJ_2151_种树_贪心+堆+链表
BZOJ_2151_种树_贪心+堆 Description A城市有一个巨大的圆形广场,为了绿化环境和净化空气,市政府决定沿圆形广场外圈种一圈树.园林部门得到指令后,初步规划出n个种树的位置,顺时针编 ...
- [luogu3620][APIO/CTSC 2007]数据备份【贪心+堆+链表】
题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏 ...
- [bzoj2288][pojChallenge]生日礼物【贪心+堆+链表】
题目描述 ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, -, AN. 她被允许选择不超过 M 个连续的部分作为自己的生日礼物. 自然地,ftiasch想要知 ...
随机推荐
- vs2008与windbg的使用
WinDbg是微软发布的一款相当优秀的源码级(source-level)调试工具,可以用于Kernel模式调试和用户模式调试,还可以调试Dump文件.使用windgb进行程序调试是一个很有必要的技能. ...
- Unity3d《Shader篇》Logo闪光特效
Shader "Custom/Flash" { Properties { _MainTex ("Base (RGB)", 2D) = "white&q ...
- ACM/ICPC 之 分治法入门(画图模拟:POJ 2083)
题意:大致就是要求画出这个有规律的Fractal图形了= = 例如 1 对应 X 2 对应 X X X X X 这个题是个理解分治法很典型的例子(详情请参见Code) 分治法:不断缩小规 ...
- AOP理解
1.问题 问题:想要添加日志记录.性能监控.安全监测 2.最初解决方案 2.1.最初解决方案 缺点:太多重复代码,且紧耦合 2.2.抽象类进行共性设计,子类进行个性设计,此处不讲解,缺点一荣俱荣,一损 ...
- 页面上有两个元素id相同,js中如何取值
页面上有两个table,id都是”cont2",现要在js中取到这两个table,改变样式. js实现: var tab2=document.all.cont2(1);var tab=do ...
- 【XLL API 函数】xlStack
查看堆栈区还剩余多少空间 原型 Excel12(xlStack, LPXLOPER12 pxRes, 0); 参数 此函数没有带任何参数 属性值/返回值 返回堆栈区还剩余的字节数 备注 返回最新版本的 ...
- python基础——调试
python基础——调试 程序能一次写完并正常运行的概率很小,基本不超过1%.总会有各种各样的bug需要修正.有的bug很简单,看看错误信息就知道,有的bug很复杂,我们需要知道出错时,哪些变量的值是 ...
- Navicat 回复 psc 文件 Mysql
在mysql 中回复 psc文件 的时候 只能一步步来,先在navicat中建一个空数据库,然后点击有上角的备份==>回复备份==> 找到psc文件==> 注意此时不要急于点击 开始 ...
- poj1988(并查集)
题目链接:http://poj.org/problem?id=1988 题意:有n个箱子,初始时每个箱子单独为一列: 接下来有p行输入,M, x, y 或者 C, x: 对于M,x,y:表示将x箱子所 ...
- NYOJ926(概率)
题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=926 设最终A获胜的概率为P,则B获胜的概率为1-P: 因此我们只需要考虑A获胜的概率即可 ...