【BZOJ2138】stone
好厉害的题啊
这道题不难看成一个二分图模型,但是给人一种求最大匹配的感觉,这实在不是很好求的样子,于是自闭了
但是不妨这样来考虑,对于一个需求\(k_i\),我们求一个最大的\(x\leq k_i\),使得这张图存在完美匹配就好了,这样我们就能愉快的使用hall定理了
我们把所有区间排个序,由于保证了没有包含关系,所以左、右端点都是单调的;
众所周知hall定理要求的是子集,但子集显然不是很好求的样子;而这个题的特殊性质可以使我们只考虑区间的情况,至于为什么,还是比较显然的。
设第\(i\)次实际上丢掉了\(b_i\)个石子,那么为了存在完美匹配,需要时刻满足\(\sum_{i=l}^rb_i\leq \sum_{i=fl[l]}^{fr[r]}a_i\),利用前缀和把这个狮子打开,即\(B_r-B_{l-1}\leq A_{fr[r]}-A_{fl[l]-1}\),即\(B_r-A_{fr[r]}\leq B_{l-1}-A_{fl[l]-1}\)
假设第\(i\)操作排序之后在第\(j\)个位置,那么我们如果要使得这次操作丢出\(x\)个石子,必须满足\(\forall k\geq j,t< j,B_{k}+x-A_{fr[k]}\leq B_t-A_{fl[t]-1}\),即\(x\leq \min(B_t-A_{fl[t]-1})-\max(B_{k}-A_{fr[k]})\)
于是我们维护两棵线段树,支持后缀min,前缀max以及区间加就好了
代码
#include<bits/stdc++.h>
#define re register
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9')c=getchar();
while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const int maxn=5e4+5;
struct Node{int l,r,t;}q[maxn];
inline int cmp(const Node A,const Node B) {return A.l<B.l;}
int n,X,Y,Z,P,c[maxn],b[maxn],A[maxn],pre[maxn],m,K[maxn],ti[maxn],cnt,G[maxn];
inline int max(int a,int b) {return a>b?a:b;}
inline int min(int a,int b) {return a<b?a:b;}
struct Segment_Tree {
int l[maxn<<2],r[maxn<<2],mx[maxn<<2],mn[maxn<<2],tag[maxn<<2];
inline void add(int v,int i) {tag[i]+=v;mx[i]+=v,mn[i]+=v;}
inline void pushup(int i) {
mx[i]=max(mx[i<<1],mx[i<<1|1]);
mn[i]=min(mn[i<<1],mn[i<<1|1]);
}
inline void pushdown(int i) {
if(!tag[i]) return;
add(tag[i],i<<1);add(tag[i],i<<1|1);tag[i]=0;
}
void build(int x,int y,int i) {
l[i]=x,r[i]=y;
if(x==y) {
mx[i]=mn[i]=G[x];
return;
}
int mid=x+y>>1;
build(x,mid,i<<1),build(mid+1,y,i<<1|1);
pushup(i);
}
void change(int x,int y,int v,int i) {
if(x<=l[i]&&y>=r[i]) {add(v,i);return;}
pushdown(i);int mid=l[i]+r[i]>>1;
if(x<=mid) change(x,y,v,i<<1);
if(y>mid) change(x,y,v,i<<1|1);
pushup(i);
}
int query_mn(int x,int y,int i) {
if(x<=l[i]&&y>=r[i]) return mn[i];
pushdown(i);int mid=l[i]+r[i]>>1;
if(y<=mid) return query_mn(x,y,i<<1);
if(x>mid) return query_mn(x,y,i<<1|1);
return min(query_mn(x,y,i<<1),query_mn(x,y,i<<1|1));
}
int query_mx(int x,int y,int i) {
if(x<=l[i]&&y>=r[i]) return mx[i];
pushdown(i);int mid=l[i]+r[i]>>1;
if(y<=mid) return query_mx(x,y,i<<1);
if(x>mid) return query_mx(x,y,i<<1|1);
return max(query_mx(x,y,i<<1),query_mx(x,y,i<<1|1));
}
}T[2];
int main() {
n=read(),X=read(),Y=read(),Z=read(),P=read();
for(re long long i=1;i<=n;i++)
A[i]=(1ll*(i-X)*(i-X)%P+1ll*(i-Y)*(i-Y)%P+1ll*(i-Z)*(i-Z)%P)%P;
m=read(),K[1]=read(),K[2]=read(),X=read(),Y=read(),Z=read(),P=read();
for(re int i=3;i<=m;i++) K[i]=(1ll*K[i-1]*X%P+1ll*K[i-2]*Y%P+Z)%P;
for(re int i=1;i<=m;i++) q[i].l=read(),q[i].r=read(),q[i].t=i;
if(!m) return 0;
std::sort(q+1,q+m+1,cmp);
for(re int i=1;i<=m;i++) c[q[i].l]++,c[q[i].r+1]--,ti[q[i].t]=i;
for(re int i=1;i<=n;i++) {
c[i]+=c[i-1];
if(c[i]) b[++cnt]=i;
}
for(re int i=1;i<=cnt;i++) pre[i]=pre[i-1]+A[b[i]];
for(re int i=1;i<=m;i++)
q[i].l=std::lower_bound(b+1,b+cnt+1,q[i].l)-b,q[i].r=std::lower_bound(b+1,b+cnt+1,q[i].r)-b;
for(re int i=1;i<=m;i++) G[i]=-pre[q[i].r];T[0].build(1,m,1);
for(re int i=0;i<m;i++) G[i]=-pre[q[i+1].l-1];T[1].build(0,m-1,1);
for(re int i=1;i<=m;i++) {
int x=ti[i],nw;
nw=T[1].query_mn(0,x-1,1)-T[0].query_mx(x,m,1);
nw=min(K[i],nw);
T[0].change(x,m,nw,1);T[1].change(x,m-1,nw,1);
printf("%d\n",nw);
}
}
【BZOJ2138】stone的更多相关文章
- 【BZOJ2138】stone Hall定理+线段树
[BZOJ2138]stone Description 话说Nan在海边等人,预计还要等上M分钟.为了打发时间,他玩起了石子.Nan搬来了N堆石子,编号为1到N,每堆包含Ai颗石子.每1分钟,Nan会 ...
- 【BZOJ2138】stone(线段树,Hall定理)
[BZOJ2138]stone(线段树,Hall定理) 题面 BZOJ 题解 考虑一个暴力. 我们对于每堆石子和每个询问,显然是匹配的操作. 所以可以把石子拆成\(a_i\)个,询问点拆成\(K_i\ ...
- 【BZOJ2138】stone(线段树+hall定理)
传送门 题意: 现在有\(n\)堆石子,每堆石子有\(a_i\)个. 之后会有\(m\)次,每次选择\([l,r]\)的石子堆中的石子扔\(k\)个,若不足,则尽量扔. 现在输出\(1\)~\(m\) ...
- 【LeetCode】动态规划(下篇共39题)
[600] Non-negative Integers without Consecutive Ones [629] K Inverse Pairs Array [638] Shopping Offe ...
- 【LeetCode】数学(共106题)
[2]Add Two Numbers (2018年12月23日,review) 链表的高精度加法. 题解:链表专题:https://www.cnblogs.com/zhangwanying/p/979 ...
- 【UVA1378】A Funny Stone Game (博弈-求SG值-输出方案)
[题目] Description The funny stone game is coming. There are n piles of stones, numbered with 0, 1, 2, ...
- HDU 5874 Friends and Enemies 【构造】 (2016 ACM/ICPC Asia Regional Dalian Online)
Friends and Enemies Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Othe ...
- 【POJ1113】Wall(凸包)
[题目] Description Once upon a time there was a greedy King who ordered his chief Architect to build a ...
- 第一阶段——CentOS6_Python3.6.1笔记(尚学堂-Python基础快速入门)+ 【补充】麦子-Python程序入门与进阶
虚拟机环境: 设置网络 .修改网络地址 .设置网卡为nat模式 .确保物理机启动dhcp.net服务 .编辑文件:vim /etc/sysconfig/network-scripts/ifcfg-et ...
随机推荐
- 【记录】mysql 查看某数据库各个表容量大小SQL
有时候我们可能需要了解数据库中各个表的容量及大小,好了解数据库情况. 现只需要一条SQL就可以直观的展示出来. SELECT table_schema AS '数据库', table_name AS ...
- 左上角小猫猫直达博主GitHub \-_-/
GitHub上有博主代码工程学习笔记啥的,由于推送比较方便所以有些学习笔记就没有上传到博客园
- java中的进制转换以及字符串类和数值类的相互转化
import java.util.*; import java.io.*; import java.math.*; import java.math.*; public class Main { pu ...
- springboot+jsp项目实例(第二弹)(成功)
1.创建spring boot项目,使用idea自带的spring initializr创建Spring boot的maven项目(我是先创建了一个空的项目). 开始创建Spring boot项目,点 ...
- note2
- java获取当前月第一天和最后一天
获取当前月第一天: /** * 获取当前月第一天 * @param month * @return */ public static String getFirstDayOfMonth(int mon ...
- 日志数据如何同步到MaxCompute
摘要:日常工作中,企业需要将通过ECS.容器.移动端.开源软件.网站服务.JS等接入的实时日志数据进行应用开发.包括对日志实时查询与分析.采集与消费.数据清洗与流计算.数据仓库对接等场景.本次分享主要 ...
- UIView响应事件的两个方法
参考自:https://blog.csdn.net/mushaofeng1990/article/details/62434349 用户触摸屏幕后的事件传递过程: //方法A-(UIView *)hi ...
- C#将Json字符串转化为对象
实体类: public class CheckData { public string msg; public string code; public string data; public stri ...
- 26 October in 614
Practice tower 有 \(N\,(2\le N\le 600000)\) 块砖,要搭一个 \(N\) 层的塔,要求:如果砖 \(A\) 在砖 \(B\) 上面,那么 \(A\) 不能比 \ ...