[BZOJ 3218]a + b Problem
又是一道主席树优化网络流的好题
按约大爷的教导,源点为白,汇点为黑,搞成最小割
发现暴力连边要爆炸,但是要连的点在线段树中都构成了一个区间,果断主席树优化之
为什么不用一般线段树?
因为要满足 j<i ,这里的可持久化并不是为了查询过去的值,而是为了保留过去的值不与后来弄混~
如果有两个点的 a[i] 相同在线段树里怎么搞?
很简单,从 a[i] 向 a[j] 连一条 inf 的流即可
不过——为什么题目名字那么坑啊啊啊啊啊啊?!!!!!
这种题是不是非要来卡空间不然不痛快是吗?!!!!!
#include <cstdio>
#include <cstring>
const int sizeOfSegment=;
const int sizeOfEdge=;
const int sizeOfPoint=;
const int sizeOfCell=;
const int inf=; inline int min(int, int);
inline int getint();
inline void putint(int); int S, T;
int V;
int n;
int a[sizeOfCell], b[sizeOfCell], w[sizeOfCell], l[sizeOfCell], r[sizeOfCell], p[sizeOfCell]; struct edge {int point, flow; edge * next, * pair;};
edge memory_edge[sizeOfEdge], * port_edge=memory_edge;
edge * e[sizeOfPoint];
inline edge * newedge(int, int, edge * );
inline void link(int, int, int);
int h[sizeOfPoint], gap[sizeOfPoint];
inline bool bfs();
inline int isap(); struct seg {int p; seg * l, * r;};
seg memory_seg[sizeOfSegment], * port_seg=memory_seg;
seg * t;
inline seg * newseg(seg * =NULL);
seg * insert(seg * , int, int, int, int);
void query(seg * , int, int, int, int, int); int main()
{
int ans=; n=getint();
S=; T=n+; V=n+;
for (int i=;i<=n;i++)
{
a[i]=getint(), b[i]=getint(), w[i]=getint(), l[i]=getint(), r[i]=getint(), p[i]=getint();
ans+=b[i]+w[i];
link(S, i, b[i]); link(i, T, w[i]);
link(i, ++V, p[i]);
query(t, , inf, l[i], r[i], V);
t=insert(t, , inf, a[i], i);
} ans-=isap();
putint(ans); return ;
}
inline int min(int x, int y)
{
return x<y?x:y;
}
inline int getint()
{
register int num=;
register char ch;
do ch=getchar(); while (ch<'' || ch>'');
do num=num*+ch-'', ch=getchar(); while (ch>='' && ch<='');
return num;
}
inline void putint(int num)
{
char stack[];
register int top=;
if (num==) stack[top=]='';
for ( ;num;num/=) stack[++top]=num%+'';
for ( ;top;top--) putchar(stack[top]);
putchar('\n');
}
inline edge * newedge(int point, int flow, edge * next)
{
edge * ret=port_edge++;
ret->point=point; ret->flow=flow; ret->next=next;
return ret;
}
inline void link(int u, int v, int f)
{
e[u]=newedge(v, f, e[u]); e[v]=newedge(u, , e[v]);
e[u]->pair=e[v]; e[v]->pair=e[u];
}
inline bool bfs()
{
static int q[sizeOfPoint];
static int l, r;
memset(h, 0xFF, sizeof(h)); h[T]=;
l=r=;
for (q[r++]=T;l<r;l++)
{
int u=q[l];
++gap[h[u]];
for (edge * i=e[u];i;i=i->next) if (h[i->point]==-)
{
h[i->point]=h[u]+;
q[r++]=i->point;
}
}
return h[S]>-;
}
inline int isap()
{
static edge * t[sizeOfPoint], * p[sizeOfPoint];
static int aug[sizeOfPoint];
int flow=, hmin=; if (!bfs()) return ; memcpy(t, e, sizeof(e)); memset(p, , sizeof(p));
aug[S]=inf;
for (int u=S;h[S]<V; )
{
if (u==T)
{
flow+=aug[T];
for (edge * i=p[T];i;i=p[i->point])
aug[i->point]-=aug[T], i->pair->flow-=aug[T], i->flow+=aug[T];
for (edge * i=p[T];i;i=p[i->point]) if (aug[i->point])
{
u=i->point;
break;
}
} edge *& i=t[u];
for ( ;i && (!i->flow || h[i->point]+!=h[u]);i=i->next);
if (i)
{
aug[i->point]=min(aug[u], i->flow); p[i->point]=i->pair;
u=i->point;
}
else
{
if (!--gap[h[u]]) break;
hmin=V;
for (edge * j=e[u];j;j=j->next) if (j->flow && h[j->point]+<hmin)
{
hmin=h[j->point]+;
t[u]=j;
}
++gap[h[u]=hmin];
u=u==S?S:p[u]->point;
}
} return flow;
}
inline seg * newseg(seg * t)
{
seg * newt=port_seg++;
newt->p=++V; newt->l=t?t->l:NULL; newt->r=t?t->r:NULL;
return newt;
}
seg * insert(seg * t, int l, int r, int p, int v)
{
seg * newt=newseg(t);
if (l==r)
{
if (t) link(newt->p, t->p, inf);
link(newt->p, v, inf);
}
else
{
int m=(l+r)>>;
if (p<=m)
{
newt->l=insert(newt->l, l, m, p, v), link(newt->p, newt->l->p, inf);
if (newt->r) link(newt->p, newt->r->p, inf);
}
else
{
newt->r=insert(newt->r, m+, r, p, v), link(newt->p, newt->r->p, inf);
if (newt->l) link(newt->p, newt->l->p, inf);
}
} return newt;
}
void query(seg * t, int l, int r, int ql, int qr, int v)
{
if (!t) return ;
if (l==ql && r==qr) link(v, t->p, inf);
else
{
int m=(l+r)>>;
if (qr<=m) query(t->l, l, m, ql, qr, v);
else if (ql>=m+) query(t->r, m+, r, ql, qr, v);
else query(t->l, l, m, ql, m, v), query(t->r, m+, r, m+, qr, v);
}
}
改数组改到想去死
尝试着用了用 Data Display Debugger 觉得是很强大但是总觉得也很不顺手?
不过反正有了这个我是不想再去用 gdb 了……
[BZOJ 3218]a + b Problem的更多相关文章
- [BZOJ 3218] A + B Problem 【可持久化线段树 + 网络流】
题目连接:BZOJ - 3218 题目分析 题目要求将 n 个点染成黑色或白色,那么我们可以转化为一个最小割模型. 我们规定一个点 i 最后属于 S 集表示染成黑色,属于 T 集表示染成白色,那么对于 ...
- bzoj 3218 a + b Problem(最小割+主席树)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3218 [题意] 给n个格子涂白或黑色,白则wi,黑则bi的好看度,若黑格i存在: 1& ...
- BZOJ.3218.a + b Problem(最小割ISAP 可持久化线段树优化建图)
BZOJ UOJ 首先不考虑奇怪方格的限制,就是类似最大权闭合子图一样建图. 对于奇怪方格的影响,显然可以建一条边\((i\to x,p_i)\),然后由\(x\)向\(1\sim i-1\)中权值在 ...
- bzoj 3218: a + b Problem【主席树+最小割】
直接建图比较显然,是(s,i,w),(i,t,b),(i,i',p),(i,j,inf),然而建出来之后发现边数是n方级别的,显然跑不过去,然后就有一种比较神的思路:把a离散了建一棵权值线段树,然后要 ...
- BZOJ 3218 A + B Problem (可持久化线段树+最小割)
做法见dalao博客 geng4512的博客, 思路就是用线段树上的结点来进行区间连边.因为有一个只能往前面连的限制,所以还要可持久化.(duliu) 一直以来我都是写dinicdinicdinic做 ...
- 【BZOJ 3218】 3218: a + b Problem(最小割+可持久化线段树)
3218: a + b Problem Time Limit: 20 Sec Memory Limit: 40 MBSubmit: 1440 Solved: 545 Description Inp ...
- [BZOJ 2301] [HAOI 2011] Problem b (莫比乌斯反演)(有证明)
[BZOJ 2301] [HAOI 2011] Problem b (莫比乌斯反演)(有证明) 题面 T组询问,每次给出a,b,c,d,k,求\(\sum _{i=a}^b\sum _{j=c}^d[ ...
- BZOJ 3218 UOJ #77 A+B Problem (主席树、最小割)
大名鼎鼎的A+B Problem, 主席树优化最小割-- 调题死活调不对,一怒之下改了一种写法交上去A了,但是改写法之后第4,5个点常数变大很多,于是喜提UOJ全站倒数第三 目前还不知道原来的写法为什 ...
- 「BZOJ 3218」 a + b Problem
题目链接 戳我 \(Solution\) 题目为什么是\(a\ +\ b\ Problem\)啊?这和题面半毛钱关系都没有. 现在来讲一下这题的解法吧,我们首先看看没有奇怪的方格这一个条件吧. 其实没 ...
随机推荐
- Python的平凡之路(16)
一.HTML+CSS补充 0.常用页面布局 <!DOCTYPE html> <html lang="en"><head> <meta ch ...
- 【python】面向对象
面向对象编程--Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 类class 实例(i ...
- 【CMD】
1.dir 2. set (不带参数) 查看环境变量. SET [variable=[string]] variable 指定环境变量名. string 指定要指派给变量的一系列字符串. 3.
- 关于ajax跨域请求(cross Domain)
Cross Domain AJAX主要就是A.com网站的页面发出一个XMLHttpRequest,这个Request的url是B.com,这样的请求是被禁止的,浏览器处于安全考虑不允许进行跨域访问, ...
- vs2013源码编译zlib 1.2.8
1.从 zlib 官网上下载 zlib最新版 1.28 的源码,解压到 zlib-1.2.8 2.使用vs2013打开vc11目录下的sln工程文件(进行单向升级) 3.修改zlibvc工程属性--& ...
- 互联网中一些常用指标(PV、UV、蹦失率、转换率、退出率)
1) PV:PageView 页面点击量,每次刷新就算一次浏览,多次打开同一页面会累加. 通常是衡量网站的主要指标. 2)UV:Unique Visitor一天内访问网站的人数(是以cookie为 ...
- 二叉树的实现与一些基本操作(C++环境)
#include<cstdio>#include<cstdlib>#include<iostream>#include<cstring>using na ...
- CABasicAnimation的delegate的坑
博客已经迁移到 www.chjsun.top 在自定义动画的时候,CABasicAnimation用的还算的蛮多的. 在此先介绍一下CABasicAnimation怎么使用. 属性介绍 属性 说明 ...
- GitHub菜鸟日志1——20160531
好吧,事实上很早就知道有github这个东西了,然而就有一种莫名的力量一直阻止着我向这“未知的领域”涉足(which is called lazy). 然后,前略...总之,默默的就开始了github ...
- 指针数组 null与空字符串
指针数组常适用于指向若干字符串,这样使字符串处理更加灵活方便. 在c++中,null表示:对象为空,它是对指针而言的.而""表示:值为空,它是对字符串而言的.