BZOJ5326 : [Jsoi2017]博弈
将所有物品按照$b$的选择顺序排序,则先手在任意前$i$个物品中最多只能拿走$\lceil\frac{i}{2}\rceil$个物品。
将每个物品的价值设为$a+b$,那么答案为先手拿走的价值和减去所有物品的$b$之和,目标是最大化先手拿走的价值和。
如果不考虑修改,则满足拟阵,可以贪心选取,修改时最优解最多发生一处变动。
每个修改可以看作是先删除一个物品,再插入一个物品。
设$f[i]$表示$[1,i]$里选择的物品数减去$\lceil\frac{i}{2}\rceil$,则方案合法的条件为$\max(f[1,n])\leq 0$。
对于插入物品$x$:
- 如果$x$可以直接加入最优解,那么直接加入。
- 否则加入$x$会导致$f$中出现$1$,那么需要找到一个最优解中的价值最小的物品$y$,满足$y$在$f$从左往右第一个$1$之前,要么不选$x$,要么将$y$替换成$x$。
对于删除物品$x$:
- 如果$x$不在最优解中,那么直接删除。
- 否则删除$x$后可以继续多加入一个物品,那么需要找到一个不在最优解中的价值最大的物品$y$,满足$y$到$n$的所有$f$都是负数。
以上所有操作都可以用线段树维护,时间复杂度$O((n+m)\log n)$。
#include<cstdio>
#include<algorithm>
using namespace std;
typedef pair<int,int>P;
const int N=100010,M=262150,inf=~0U>>1,BUF=10000000;
char Buf[BUF],*buf=Buf;long long ans;
int n,m,i,x,y,pos[N];bool used[N];
P ma[M],mi[M];int f[M],tag[M];
struct E{int a,b,p;}e[N];
inline bool cmp(const E&a,const E&b){
if(a.b!=b.b)return a.b>b.b;
return a.p<b.p;
}
inline void read(int&a){for(a=0;*buf<48;buf++);while(*buf>47)a=a*10+*buf++-48;}
inline void up(int x){
ma[x]=max(ma[x<<1],ma[x<<1|1]);
mi[x]=min(mi[x<<1],mi[x<<1|1]);
}
void build(int x,int a,int b){
if(a==b){
ma[x]=P(-inf,a);
mi[x]=P(inf,a);
f[x]=-(a+1)/2;
return;
}
int mid=(a+b)>>1;
build(x<<1,a,mid),build(x<<1|1,mid+1,b),up(x);
f[x]=max(f[x<<1],f[x<<1|1]);
}
inline void tag1(int x,int p){f[x]+=p;tag[x]+=p;}
inline void pb(int x){if(tag[x])tag1(x<<1,tag[x]),tag1(x<<1|1,tag[x]),tag[x]=0;}
void maketag(int x,int a,int b,int c,int d,int p){
if(c<=a&&b<=d){tag1(x,p);return;}
pb(x);
int mid=(a+b)>>1;
if(c<=mid)maketag(x<<1,a,mid,c,d,p);
if(d>mid)maketag(x<<1|1,mid+1,b,c,d,p);
f[x]=max(f[x<<1],f[x<<1|1]);
}
void setused(int x,int a,int b,int c,int p){
if(a==b){
used[a]=1;
ma[x]=P(-inf,a);
mi[x]=P(p,a);
return;
}
int mid=(a+b)>>1;
if(c<=mid)setused(x<<1,a,mid,c,p);else setused(x<<1|1,mid+1,b,c,p);
up(x);
}
void setunused(int x,int a,int b,int c,int p){
if(a==b){
used[a]=0;
ma[x]=P(p,a);
mi[x]=P(inf,a);
return;
}
int mid=(a+b)>>1;
if(c<=mid)setunused(x<<1,a,mid,c,p);else setunused(x<<1|1,mid+1,b,c,p);
up(x);
}
int askf(int x,int a,int b,int c,int d){
if(c<=a&&b<=d)return f[x];
pb(x);
int mid=(a+b)>>1,t=-inf;
if(c<=mid)t=askf(x<<1,a,mid,c,d);
if(d>mid)t=max(t,askf(x<<1|1,mid+1,b,c,d));
return t;
}
P askmi(int x,int a,int b,int c,int d){
if(c<=a&&b<=d)return mi[x];
int mid=(a+b)>>1;P t(inf,0);
if(c<=mid)t=askmi(x<<1,a,mid,c,d);
if(d>mid)t=min(t,askmi(x<<1|1,mid+1,b,c,d));
return t;
}
P askma(int x,int a,int b,int c,int d){
if(c<=a&&b<=d)return ma[x];
int mid=(a+b)>>1;P t(-inf,0);
if(c<=mid)t=askma(x<<1,a,mid,c,d);
if(d>mid)t=max(t,askma(x<<1|1,mid+1,b,c,d));
return t;
}
inline int findfirstpos(){
int x=1,a=1,b=n,mid;
while(a<b){
pb(x);
mid=(a+b)>>1;
if(f[x<<1]>0)x=x<<1,b=mid;else x=x<<1|1,a=mid+1;
}
return a;
}
inline int findlastneg(){
int x=1,a=1,b=n,mid,ret=n+1;
while(a<b){
pb(x);
mid=(a+b)>>1;
if(f[x<<1|1]<0){
x=x<<1;
b=mid;
ret=mid+1;
}else{
x=x<<1|1;
a=mid+1;
}
}
if(f[x]<0)ret=a;
return ret;
}
inline void additem(int x){
int val=e[x].a+e[x].b,tmp=askf(1,1,n,x,n);
maketag(1,1,n,x,n,1);
if(tmp<0){
setused(1,1,n,x,val);
ans+=val;
return;
}
int o=findfirstpos();
maketag(1,1,n,x,n,-1);
P y=askmi(1,1,n,1,o);
if(y.first>=val){
setunused(1,1,n,x,val);
return;
}
ans+=val-y.first;
setunused(1,1,n,y.second,y.first);
maketag(1,1,n,y.second,n,-1);
setused(1,1,n,x,val);
maketag(1,1,n,x,n,1);
}
inline void delitem(int x){
if(!used[x])return;
ans-=e[x].a+e[x].b;
setunused(1,1,n,x,-inf);
maketag(1,1,n,x,n,-1);
int o=findlastneg();
if(o>n)return;
P y=askma(1,1,n,o,n);
if(y.first<=0)return;
ans+=y.first;
setused(1,1,n,y.second,y.first);
maketag(1,1,n,y.second,n,1);
}
int main(){
fread(Buf,1,BUF,stdin);read(n);
for(i=1;i<=n;i++)read(e[i].a);
for(i=1;i<=n;i++)read(e[i].b),e[i].p=i,ans-=e[i].b;
sort(e+1,e+n+1,cmp);
for(i=1;i<=n;i++)pos[e[i].p]=i;
build(1,1,n);
for(i=1;i<=n;i++)additem(i);
printf("%lld\n",ans);
read(m);
while(m--){
read(x),read(y);
x=pos[x];
delitem(x);
e[x].a=y;
additem(x);
printf("%lld\n",ans);
}
return 0;
}
BZOJ5326 : [Jsoi2017]博弈的更多相关文章
- BZOJ 5326 [JSOI2017]博弈 (模拟费用流、线段树)
题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=5326 题解 终于成为第8个A掉这题的人--orz tzw神仙早我6小时 本以为这东西常数 ...
- hdu----(1849)Rabbit and Grass(简单的尼姆博弈)
Rabbit and Grass Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- HDU 5754 Life Winner Bo 组合博弈
Life Winner Bo Problem Description Bo is a "Life Winner".He likes playing chessboard gam ...
- HDU 2509 Nim博弈变形
1.HDU 2509 2.题意:n堆苹果,两个人轮流,每次从一堆中取连续的多个,至少取一个,最后取光者败. 3.总结:Nim博弈的变形,还是不知道怎么分析,,,,看了大牛的博客. 传送门 首先给出结 ...
- HDU 1907 Nim博弈变形
1.HDU 1907 2.题意:n堆糖,两人轮流,每次从任意一堆中至少取一个,最后取光者输. 3.总结:有点变形的Nim,还是不太明白,盗用一下学长的分析吧 传送门 分析:经典的Nim博弈的一点变形. ...
- 51nod1072(wythoff 博弈)
题目链接: http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1072 题意: 中文题诶~ 思路: 博弈套路是有的, 找np局 ...
- ACM: NBUT 1107 盒子游戏 - 简单博弈
NBUT 1107 盒子游戏 Time Limit:1000MS Memory Limit:65535KB 64bit IO Format: Practice Appoint ...
- 【转】ACM博弈知识汇总
博弈知识汇总 转自:http://www.cnblogs.com/kuangbin/archive/2011/08/28/2156426.html 有一种很有意思的游戏,就是有物体若干堆,可以是火柴棍 ...
- Uva 10891 经典博弈区间DP
经典博弈区间DP 题目链接:https://uva.onlinejudge.org/external/108/p10891.pdf 题意: 给定n个数字,A和B可以从这串数字的两端任意选数字,一次只能 ...
随机推荐
- 关于js dtGrid报错长度的问题
错误js截图 Uncaught TypeError: Cannot read property 'length' of undefined 翻译:Uncaught TypeError:无法读取未定义的 ...
- cmd切换代码页,切换字体,cmd不能输入中文
cmd终端切换编码:437:美国英语.936:中文gbk编码.65001:UTF8 小知识: 如果cmd不能调用中文输入法,也就是不能输入中文,是因为目前激活的代码页不是936 使用 chcp 936 ...
- 移动文件(git mv)
使用git mv命令将mian.c移动为main2.c $ git mv main.c main2.c D:\Git\test (master -> origin) $ git status O ...
- php、apache、nginx、线程、进程
最近在学swoole,发现里面设计好多操作系统里面的概念,这些基础知识正是自己欠缺的.根基不牢的高楼大厦,犹如空中楼阁,随时都要崩塌,早发现早治疗哈哈^_^. 一.概念 1) 进程:是指正在运行的一个 ...
- 查询SQL Server执行过的SQL语句
SELECT TOP 1000 ST.text AS '执行的SQL语句', QS.execution_count AS '执行次数', QS.total_ ...
- mysql 8.0~MGR多成员读一致性
一 背景:当在读节点多成员查询时可能导致数据不一致 二 三种场景 1 读多写少 AFTER 2 读写相当 AFTER_AND_BEFORE 3 读少写多 BEFORE三 数据不一致 ...
- Beta 冲刺(6/7)
目录 摘要 团队部分 个人部分 摘要 队名:小白吃 组长博客:hjj 作业博客:beta冲刺(6/7) 团队部分 后敬甲(组长) 过去两天完成了哪些任务 ppt制作 视频拍摄 接下来的计划 准备答辩 ...
- AttributeError: 'module' object has no attribute 'enableTrace'
Traceback (most recent call last): File "Long-lived-connection.py", line 29, in <module ...
- iView -- TimePicker 自定义修改时间选择器选择时间面板样式
iView官方组件展示效果: 期望的最终效果: 为什么要修改期望效果? 项目需要只选择小时,分钟跟秒的不需要,而官方并没有直接相关的小时组件或者是设置显示成小时或分钟或秒的时间选择器,因为自己直接修改 ...
- Windows【端口被占用,杀死想啥的端口】
windows 两步方法 netstat -ano | findstr "8080" taskkill /pid 4136-t -f linux 两步方法 ps -ef | gre ...