题目链接:奇怪的计算器

  如果没有溢出的话,所有的标记都可以在线段树上直接维护,所以一棵线段树就解决问题了。

  现在有了溢出,怎么办呢?

  发现就算溢出了,各个元素的相对大小关系也是不变的。所以,如果一开始就把元素排好序,溢出的数一定是两段区间。在线段树上把这两段区间找出来,区间赋值就好了。当然也需要多记录区间\(\max\)和区间\(\min\)。

  PS:注意各个标记的先后关系

  下面贴代码(代码较丑):

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)
#define maxn 100010
#define MAXN maxn<<2 using namespace std;
typedef long long llg; struct data{
int x,b;
bool operator < (const data &h)const{return x<h.x;}
}b[maxn],s[maxn];
int n,m,ZL,ZR,a[maxn],ans[maxn];
llg maxv[MAXN],minv[MAXN],che[MAXN];
llg add1[MAXN],add2[MAXN],set[MAXN]; int getint(){
int w=0;bool q=0;
char c=getchar();
while((c>'9'||c<'0')&&c!='-') c=getchar();
if(c=='-') c=getchar(),q=1;
while(c>='0'&&c<='9') w=w*10+c-'0',c=getchar();
return q?-w:w;
} void gch(int u,int x){minv[u]*=x,maxv[u]*=x,che[u]*=x;}
void ga1(int u,int x){minv[u]+=x,maxv[u]+=x,add1[u]+=x;}
void gset(int u,int x){
set[u]=maxv[u]=minv[u]=x;
add1[u]=add2[u]=0; che[u]=1;
} void pushdown(int u,int l,int r){
int lc=u<<1,lv=u<<1|1,mid=(l+r)>>1;
if(set[u]) gset(lc,set[u]),gset(lv,set[u]);
if(che[u]>1) gch(lc,che[u]),gch(lv,che[u]);
if(add1[u]) ga1(lc,add1[u]),ga1(lv,add1[u]);
if(add2[u]){
minv[lc]+=add2[u]*a[l]; maxv[lc]+=add2[u]*a[mid];
minv[lv]+=add2[u]*a[mid+1],maxv[lv]+=add2[u]*a[r];
add2[lc]+=add2[u],add2[lv]+=add2[u];
}
set[u]=add1[u]=add2[u]=0; che[u]=1;
} void update(int u){minv[u]=minv[u<<1],maxv[u]=maxv[u<<1|1];}
void build(int u,int l,int r){
int lc=u<<1,lv=u<<1|1,mid=(l+r)>>1; che[u]=1;
if(l==r){maxv[u]=minv[u]=a[l];return;}
build(lc,l,mid),build(lv,mid+1,r);
update(u);
} void queryl(){
if(maxv[1]<=ZL){gset(1,ZL);return;}
int u=1,l=1,r=m,mid;
while(l!=r){
pushdown(u,l,r);
mid=(l+r)>>1; u<<=1;
if(maxv[u]>=ZL) r=mid;
else gset(u,ZL),l=mid+1,u^=1;
}
while(u>1) u>>=1,update(u);
} void queryr(){
if(minv[1]>=ZR){gset(1,ZR);return;}
int u=1,l=1,r=m,mid;
while(l!=r){
pushdown(u,l,r);
mid=(l+r)>>1; u<<=1; u|=1;
if(minv[u]<=ZR) l=mid+1;
else gset(u,ZR),r=mid,u^=1;
}
while(u>1) u>>=1,update(u);
} void dfs(int u,int l,int r){
int lc=u<<1,lv=u<<1|1,mid=(l+r)>>1;
if(l==r) ans[s[l].b]=maxv[u];
else{
pushdown(u,l,r);
dfs(lc,l,mid),dfs(lv,mid+1,r);
}
} int main(){
File("a");
n=getint(),ZL=getint(),ZR=getint();
for(int i=1;i<=n;i++){
char c=getchar();
while(c!='+' && c!='-' && c!='*' && c!='@') c=getchar();
b[i].x=getint();
if(c=='+') b[i].b=1; if(c=='-') b[i].b=2;
if(c=='*') b[i].b=3; if(c=='@') b[i].b=4;
}
m=getint();
for(int i=1;i<=m;i++) s[i].x=getint(),s[i].b=i;
sort(s+1,s+m+1);
for(int i=1;i<=m;i++) a[i]=s[i].x;
build(1,1,m);
dfs(1,1,m);
for(int i=1,x,tp;i<=n;i++){
tp=b[i].b,x=b[i].x;
if(tp==1) ga1(1,x); if(tp==2) ga1(1,-x); if(tp==3) gch(1,x);
if(tp==4) add2[1]+=x,minv[1]+=1ll*a[1]*x,maxv[1]+=1ll*a[m]*x;
queryl(); queryr();
//dfs(1,1,m);
}
dfs(1,1,m);
for(int i=1;i<=m;i++) printf("%d\n",ans[i]);
return 0;
}

BZOJ 3878 【AHOI2014】 奇怪的计算器的更多相关文章

  1. BZOJ 3878: [Ahoi2014]奇怪的计算器

    BZOJ 3878: [Ahoi2014]奇怪的计算器 标签(空格分隔): OI-BZOJ OI-线段树 Time Limit: 10 Sec Memory Limit: 256 MB Descrip ...

  2. AHOI2014 奇怪的计算器 和 HDU5306 Gorgeous Sequence

    线段树秀操作题. 奇怪的计算器 有 N 个数,一共会对这 N 个数执行 M 个指令(对没个数执行的指令都一样),每一条指令可以是以下四种指令之一:(这里 a 表示一个正整数) 加上 a 减去 a 乘以 ...

  3. BZOJ3878: [Ahoi2014&Jsoi2014]奇怪的计算器

    BZOJ3878: [Ahoi2014&Jsoi2014]奇怪的计算器 Description [故事背景] JYY有个奇怪的计算器,有一天这个计算器坏了,JYY希望你能帮助他写 一个程序来模 ...

  4. 「AHOI2014/JSOI2014」奇怪的计算器

    「AHOI2014/JSOI2014」奇怪的计算器 传送门 我拿到这题首先是懵b的,因为感觉没有任何性质... 后来经过同机房dalao的指导发现可以把所有的 \(X\) 放到一起排序,然后我们可以发 ...

  5. BZOJ 3873: [Ahoi2014]拼图

    BZOJ 3873: [Ahoi2014]拼图 标签(空格分隔): OI-BZOJ OI-DP Time Limit: 10 Sec Memory Limit: 256 MB Description ...

  6. BZOJ 3878 [AHOI&JSOI2014]奇怪的计算器 (线段树)

    题面:BZOJ传送门 洛谷传送门 线段树好题 题目保证$a$一定是正整数,容易发现计算结果是单调的 我们把询问离线,并按照从小到大排序 某次操作可能导致某些位置达到边界$L/R$ 根据单调性的结论 这 ...

  7. 2018.07.25 bzoj3878: [Ahoi2014&Jsoi2014]奇怪的计算器(线段树)

    传送门 线段树综合. 让我想起一道叫做siano" role="presentation" style="position: relative;"&g ...

  8. AHOI2014/JSOI2014 奇怪的计算器

    题目描述 题解: 考虑到经过一系列变化后小数不可能比大数大,我们可以用线段树维护区间修改. 重点是,每个节点都可以通过$a[i]=a[i]*t1+a0[i]*t2+t3$这个函数来表示,我们就可以把三 ...

  9. BZOJ 2756: [SCOI2012]奇怪的游戏 [最大流 二分]

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 3352  Solved: 919[Submit][Stat ...

随机推荐

  1. Music life

    some songs: you are you are <<你眼中的世界> Trouble I'm In > 1:40 secs 忧桑钢琴曲 < 豆花之歌 The Tru ...

  2. HDU 1879 继续畅通工程(最小生成树)

    省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).现得到城镇道路统计表,表中列出了任意两城镇间修建道路的费用,以及该道路是否已经 ...

  3. Autel MaxiSys Elite Diagnostic Tool Common problem solving methods

    1. updating MaxiFlash Elite to firmware 3.21? My maxisys communicate with the MaxiFlash J2534 but Ma ...

  4. springMVC之一(页面<--->控制器 互相传值,转发和重定向)

    #页面--->控制器1.request:不建议使用2.使用属性传值(建议使用)@RequestParam("name") String username3.使用Bean对象传 ...

  5. php脚本#!/usr/bin/env php写法的好处

    脚本语言的第一行,目的就是指出,你想要你的这个文件中的代码用什么可执行程序去运行它.比如php脚本的第一行可以写成如下几种格式#!/usr/bin/php#!/usr/bin/env php#!/us ...

  6. Fiddler(三)Fiddler设置手机抓包

    一.前提 我们要实现手机抓包,必须要手机连接的wifi和PC段连接的wifi所处同一个局域网内,如果你使用的是笔记本,那么这个就好办了,如果你使用的是台式机,那么你还需要准备一个无线网卡.我使用的是F ...

  7. P3157 [CQOI2011]动态逆序对(树状数组套线段树)

    P3157 [CQOI2011]动态逆序对 树状数组套线段树 静态逆序对咋做?树状数组(别管归并QWQ) 然鹅动态的咋做? 我们考虑每次删除一个元素. 减去的就是与这个元素有关的逆序对数,介个可以预处 ...

  8. 创建docker镜像的私有仓库

    CentOS Linux release 7.2.1511 Docker version 17.03.1-ce 安装registry镜像 同时安装一个比较小的镜像alpine待会作测试用: # doc ...

  9. Stanford CS231n实践笔记(课时14卷积神经网络详解 上)

    本课我们主要来研究一个"浏览器中的卷积神经网络" 这只是一个展示项目,但是能够帮助直观地看到一些东西 地址:https://cs.stanford.edu/people/karpa ...

  10. java常用类-StringBuffer,Integer,Character

    * StringBuffer: * 线程安全的可变字符串. * * StringBuffer和String的区别? * 前者长度和内容可变,后者不可变. * 如果使用前者做字符串的拼接,不会浪费太多的 ...