【HNOI】d 最小割
【题目大意】给定一个n*m的土地,每块可以种a或b作物,每种作物在不同的位置有不同的收成,同时,有q个子矩阵中,全部种指定的作物(a或b)会有一定的加成收成,求最大收成。
【数据范围】
50% n,m<=10 q<=500
100% n,m<=100 q<=50000
首先我们解决小范围数据,比较容易的可以看出来这是一个最小割模型,先将ans+=value。我们只需要(source,i,value[i][0]),表示不种植a的代价,(i,sink,value[i][1])表示不种植b的代价。对于额外的加成,如果全是b作物,我们可以表示为(x,cur,inf),(cur,sink,value) x为矩阵中的所有点,这个表示我们只要矩阵中的任意一个元素没有种植b(也就是某个点割得与sink相连的边),那么我们都可以找到一条新的增广路,流量为value。
那么我们可以发现,这种建模的边是n*m*q级别的,因为每次我们新的cur点都与矩阵中所有的点连接了,我们需要来增加图的点的数量来减少边的数量,那么我们可以用二维st表来表示每个矩阵中的点,num[i][j][p][q]表示矩阵中i,j点为左上角,长为2^p,宽为2^q的矩阵,我们将图拆为a,b两层,分别表示a,b作物的矩阵。
因为新加入的点是为了简化之前的图的,所以我们同层的st表之间的边应该与之前连接的边的方向相同,因为我们这样做相当与把原图拆成了两部分,原图的两部分之间是互通的,所以我们也应该将两层st表之间加上双向边,如果不加这个的话,会出现由于额外价值过大导致割了连接源和汇的边而保留两个附加收成的点,这样当然是不合法的。
反思:开始建的是正方形的st表,后来发现了这种建发的诸多不便,这样不能保证图的规模,因为条形的矩阵可以卡掉这个。然后开始的图没有两层之间双向连边,所以导致了些奇奇怪怪的问题(也不奇怪,就是上述的不合法割边)。
//By BLADEVIL
#include <cstdio>
#include <cstring>
#include <algorithm>
#define maxm 2000010
#define maxn 110
#define inf (1000000000) using namespace std; int n,m,query;
int source,sink,tot,l;
int key[maxn][maxn][],num[maxn][maxn][][][];
int pre[maxm],other[maxm],last[maxm],len[maxm];
int que[maxm],dis[maxm]; void connect(int x,int y,int z) {
pre[++l]=last[x];
last[x]=l;
other[l]=y;
len[l]=z;
//printf("|%d %d %d\n",x,y,z);
} bool bfs() {
memset(dis,,sizeof dis);
que[]=source; dis[source]=;
int h=,t=;
while (h<t) {
int cur=que[++h];
for (int p=last[cur];p;p=pre[p]) {
if (len[p]<=) continue;
if (!dis[other[p]]) {
que[++t]=other[p];
dis[other[p]]=dis[cur]+;
if (other[p]==sink) return true;
}
}
}
return false;
} int dinic(int x,int flow) {
//printf("%d %d\n",x,flow);
if (x==sink) return flow;
int rest=flow;
for (int p=last[x];p;p=pre[p]) {
if (len[p]<=) continue;
if (!rest) continue;
if (dis[other[p]]!=dis[x]+) continue;
int tmp=dinic(other[p],min(rest,len[p]));
len[p]-=tmp; len[p^]+=tmp; rest-=tmp;
}
return flow-rest;
} int main() {
freopen("d.in","r",stdin); freopen("d.out","w",stdout);
scanf("%d%d%d",&n,&m,&query);
for (int i=;i<=n;i++)
for (int j=;j<=m;j++) scanf("%d",&key[i][j][]);
for (int i=;i<=n;i++)
for (int j=;j<=m;j++) scanf("%d",&key[i][j][]);
l=;
for (int p=;(<<p)<=n;p++)
for (int q=;(<<q)<=m;q++)
for (int i=;i+(<<p)-<=n;i++)
for (int j=;j+(<<q)-<=m;j++) {
num[i][j][p][q][]=++tot; num[i][j][p][q][]=++tot;
}
source=++tot; sink=++tot;
int ans=;
for (int i=;i<=n;i++)
for (int j=;j<=m;j++) {
connect(source,num[i][j][][][],key[i][j][]); connect(num[i][j][][][],source,);
connect(num[i][j][][][],sink,key[i][j][]); connect(sink,num[i][j][][][],);
ans+=key[i][j][]+key[i][j][];
}
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
connect(num[i][j][][][],num[i][j][][][],inf),connect(num[i][j][][][],num[i][j][][][],);
for (int p=;(<<p)<=n;p++)
for (int q=;(<<q)<=m;q++)
for (int i=;i+(<<p)-<=n;i++)
for (int j=;j+(<<q)-<=m;j++) {
connect(num[i][j][p][q][],num[i][j][p][q][],inf);connect(num[i][j][p][q][],num[i][j][p][q][],);
if (q) {
connect(num[i][j][p][q][],num[i][j][p][q-][],inf); connect(num[i][j][p][q-][],num[i][j][p][q][],);
connect(num[i][j][p][q][],num[i][j+(<<(q-))][p][q-][],inf); connect(num[i][j+(<<(q-))][p][q-][],num[i][j][p][q][],);
connect(num[i][j][p][q-][],num[i][j][p][q][],inf); connect(num[i][j][p][q][],num[i][j][p][q-][],);
connect(num[i][j+(<<(q-))][p][q-][],num[i][j][p][q][],inf); connect(num[i][j][p][q][],num[i][j+(<<(q-))][p][q-][],);
} else
if (p) {
connect(num[i][j][p][q][],num[i][j][p-][q][],inf); connect(num[i][j][p-][q][],num[i][j][p][q][],);
connect(num[i][j][p][q][],num[i+(<<(p-))][j][p-][q][],inf); connect(num[i+(<<(p-))][j][p-][q][],num[i][j][p][q][],);
connect(num[i][j][p-][q][],num[i][j][p][q][],inf); connect(num[i][j][p][q][],num[i][j][p-][q][],);
connect(num[i+(<<(p-))][j][p-][q][],num[i][j][p][q][],inf);
connect(num[i][j][p][q][],num[i+(<<(p-))][j][p-][q][],);
}
}
while (query--) {
int x1,y1,x2,y2,w,z,q=,p=,cur=++tot; scanf("%d%d%d%d%d%d",&x1,&y1,&x2,&y2,&z,&w);
ans+=w;
while ((<<(p+))<=x2-x1+) p++;
while ((<<(q+))<=y2-y1+) q++;
if (z) {
connect(cur,sink,w); connect(sink,cur,);
connect(num[x1][y1][p][q][],cur,inf); connect(cur,num[x1][y1][p][q][],);
connect(num[x1][y2-(<<q)+][p][q][],cur,inf); connect(cur,num[x1][y2-(<<q)+][p][q][],);
connect(num[x2-(<<p)+][y1][p][q][],cur,inf); connect(cur,num[x2-(<<p)+][y1][p][q][],);
connect(num[x2-(<<p)+][y2-(<<q)+][p][q][],cur,inf); connect(cur,num[x2-(<<p)+][y2-(<<q)+][p][q][],);
} else {
connect(source,cur,w); connect(cur,source,);
connect(cur,num[x1][y1][p][q][],inf); connect(num[x1][y1][p][q][],cur,);
connect(cur,num[x1][y2-(<<q)+][p][q][],inf); connect(num[x1][y2-(<<q)+][p][q][],cur,);
connect(cur,num[x2-(<<p)+][y1][p][q][],inf); connect(num[x2-(<<p)+][y1][p][q][],cur,);
connect(cur,num[x2-(<<p)+][y2-(<<q)+][p][q][],inf); connect(num[x2-(<<p)+][y2-(<<q)+][p][q][],cur,);
}
}
while (bfs()) ans-=dinic(source,inf);
printf("%d\n",ans);
fclose(stdin); fclose(stdout);
return ;
}
【HNOI】d 最小割的更多相关文章
- BZOJ 1391: [Ceoi2008]order [最小割]
1391: [Ceoi2008]order Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 1509 Solved: 460[Submit][Statu ...
- BZOJ-2127-happiness(最小割)
2127: happiness(题解) Time Limit: 51 Sec Memory Limit: 259 MBSubmit: 1806 Solved: 875 Description 高一 ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- BZOJ3438 小M的作物(最小割)
题目 Source http://www.lydsy.com/JudgeOnline/problem.php?id=3438 Description 小M在MC里开辟了两块巨大的耕地A和B(你可以认为 ...
- 最大流-最小割 MAXFLOW-MINCUT ISAP
简单的叙述就不必了. 对于一个图,我们要找最大流,对于基于增广路径的算法,首先必须要建立反向边. 反向边的正确性: 我努力查找了许多资料,都没有找到理论上关于反向边正确性的证明. 但事实上,我们不难理 ...
- bzoj1412最小割
太羞耻了,m n写反了(主要是样例n m相等) 建图方法比较高(ji)端(chu),对于可以加栅栏的地方连上1的边,然后求最小割即可 为了让代码优(suo)美(duan),我写了一个check,避免多 ...
- 【BZOJ1497】[NOI2006]最大获利 最小割
裸的最小割,很经典的模型. 建图:要求总收益-总成本最大,那么将每条弧与源点相连,流量为成本,每个收益与汇点相连,流量为收益,然后每条弧与它所能到达的收益相连,流量为inf. 与源点相连的是未被选中的 ...
- 二分图&网络流&最小割等问题的总结
二分图基础: 最大匹配:匈牙利算法 最小点覆盖=最大匹配 最小边覆盖=总节点数-最大匹配 最大独立集=点数-最大匹配 网络流: 技巧: 1.拆点为边,即一个点有限制,可将其转化为边 BZOJ1066, ...
- CQOI 2016 不同的最小割
题目大意:一个无向图,求所有点对不同的最小割种类数 最小割最多有n-1个,这n-1个最小割构成一个最小割树 分治法寻找n-1个最小割.对于当前点集X,任选两点为ST做最小割,然后找出与S相连的所有点和 ...
随机推荐
- 《学习OpenCV》课后习题解答3
题目:(P104) 创建一个大小为100*100的三通道RGB图像.将它的元素全部置0.使用指针算法以(20,5)与(40,20)为项点绘制一个绿色平面. 解答: #include "cv. ...
- 读写INI文件操作类
详情介绍:http://zh.wikipedia.org/wiki/INI%E6%96%87%E4%BB%B6 示例: 下面是一个虚拟的程序,其INI文件有两个小节,前面的小节是用来设置拥有者的信息, ...
- 火狐浏览器(FireFox)安装Flash插件失败处理方法
最近不知道怎么了,总是嫌弃IE,可能是被网络流量监测的网址给搞得了,弄了火狐浏览器,也安装了猎豹,这里不对浏览器做评价 好多朋友安装好火狐(FireFox)的时候发现之前不是有装IE的Flash播放插 ...
- MySQL错误解决10038
[错误解决]本地计算机上的mysql服务启动停止后,某些服务在未由其他服务或程序使用时将自动停止 标签: mysql计算机 2016-12-01 17:49 5508人阅读 评论(2) 收藏 举报 ...
- BZOJ 1191 超级英雄(二分图匹配)
把题目作为s集,锦囊作为t集.把每个题目和它可以用的锦囊连边,这样就构成了一个二分图,求出这个二分图最大匹配. 但是这个最大匹配有限制条件,就是对于每个可能的匹配集,如果s集的i点有匹配,那么i-1点 ...
- 【bzoj1609】[Usaco2008 Feb]Eating Together麻烦的聚餐 dp
题目描述 为了避免餐厅过分拥挤,FJ要求奶牛们分3批就餐.每天晚饭前,奶牛们都会在餐厅前排队入内,按FJ的设想所有第3批就餐的奶牛排在队尾,队伍的前端由设定为第1批就餐的奶牛占据,中间的位置就归第2批 ...
- datepicker约束开始时间和结束时间
datepicker约束开始时间和结束时间作用就是:选择要搜索的日期范围. <!DOCTYPE html> <html lang="en"> <hea ...
- Codeforces Round #510 Div. 2 Virtual Participate记
这场打的顺手到不敢相信.如果不是vp的话估计肯定打不到这个成绩. A:最大显然,最小的话每次暴力给最小的+1. #include<iostream> #include<cstdio& ...
- NYOJ 1000 又见斐波那契数列
描述 斐波那契数列大家应该很熟悉了吧.下面给大家引入一种新的斐波那契数列:M斐波那契数列. M斐波那契数列F[n]是一种整数数列,它的定义如下: F[0] = a F[1] = b F[n] = F[ ...
- 算法训练 Bus Tour
问题描述 想象你是一个在Warsaw的游客,而且预订了一次乘车旅行,去城镇外看一些令人惊异的景点.这辆公共汽车首先围绕城镇行驶一段时间(一段很长的时间,由于Warsaw是一个大城市),把在各自旅馆的人 ...