题意

现在有一个怪兽序列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 贪心加链表的更多相关文章

  1. CF733C Epidemic in Monstropolis[模拟 构造 贪心]

    C. Epidemic in Monstropolis time limit per test 1 second memory limit per test 256 megabytes input s ...

  2. Epidemic in Monstropolis

    Epidemic in Monstropolis 题目链接:http://codeforces.com/contest/733/problem/C 贪心 新序列的m个数肯定是由原序列的连续的m个子序列 ...

  3. 【16.52%】【codeforces 733C】Epidemic in Monstropolis

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  4. 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 ...

  5. 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 ...

  6. Codeforces 733C:Epidemic in Monstropolis(暴力贪心)

    http://codeforces.com/problemset/problem/733/C 题意:给出一个序列的怪兽体积 ai,怪兽只能吃相邻的怪兽,并且只有体积严格大于相邻的怪兽才能吃,吃完之后, ...

  7. BZOJ_2151_种树_贪心+堆+链表

    BZOJ_2151_种树_贪心+堆 Description A城市有一个巨大的圆形广场,为了绿化环境和净化空气,市政府决定沿圆形广场外圈种一圈树.园林部门得到指令后,初步规划出n个种树的位置,顺时针编 ...

  8. [luogu3620][APIO/CTSC 2007]数据备份【贪心+堆+链表】

    题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏 ...

  9. [bzoj2288][pojChallenge]生日礼物【贪心+堆+链表】

    题目描述 ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, -, AN. 她被允许选择不超过 M 个连续的部分作为自己的生日礼物. 自然地,ftiasch想要知 ...

随机推荐

  1. cairo-1.14.6 static compiler msys mingw32

    gtk2.x 静态编译时 需要注意的是 cairo cairo 1.14.x 使用了 mutex , 用动态方式时 DllMain 中调用了 CAIRO_MUTEX_INITIALIZE () 在静态 ...

  2. 4.kvm克隆虚拟机

    virt-clone 作用简介 virt-clone 主要是用来克隆kvm虚拟机,并且通过 Options.General Option.Storage Configuration.Networkin ...

  3. Effective C++ -----条款27:尽量少做转型动作

    如果可以,尽量避免转型,特别是在注重效率的代码中避免dynamic_casts.如果有个设计需要转型动作,试着发展无需转型的替代设计. 如果转型是必要的,试着将它隐藏于某个函数背后.客户随后可以调用该 ...

  4. nyoj_34_韩信点兵

     中国剩余定理: 代码: #include <iostream> #include <cstdio> using namespace std; int main() { int ...

  5. HTML 基础

    1.HTML  超文本标记语言 2.网页分类: 动态网页   静态网页 ①静态网页与动态网页区别:  主要:动态网页与数据库链接,静态网页不与数据库连接: ②静态网页 修改展示图片 必须修改源代码 : ...

  6. Android Volley入门到精通:使用Volley加载网络图片

    在上一篇文章中,我们了解了Volley到底是什么,以及它的基本用法.本篇文章中我们即将学习关于Volley更加高级的用法,如何你还没有看过我的上一篇文章的话,建议先去阅读Android Volley完 ...

  7. 【leetcode】Linked List Cycle II (middle)

    Given a linked list, return the node where the cycle begins. If there is no cycle, return null. Foll ...

  8. 【leetcode】Best Time to Buy and Sell 2(too easy)

    Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...

  9. 【Git】参与github上其他人的项目

    来源:廖雪峰 访问感兴趣的项目主页.,点“Fork”就在自己的账号下克隆了该项目仓库,然后,从自己的账号下clone到本地,就可以工作啦~ 以bootstrap项目为例,这个关系如下图所示: 一定要从 ...

  10. IOS- DocumentInteraction Controllerl的使用

    iOS提供了使用其他app预览文件的支持,这就是Document Interaction Controller.此外,iOS也支持文件关联,允许其他程序调用你的app打开某种文件.而且,从4.2开始, ...