P1361 小M的作物 最小割理解
如果没有组合效益的存在 我们直接每个点两部分的最大值即可
换成网络流模型来看 即把S点看作是A田 把T点看作是B田 每种作物看作一个点 分别连边(S,i,A[i]) (i,T,B[i])
最后图中所有边权和减去最大流即为答案.这个很好理解,因为最小割=最大流,一种作物只能选择A,B里的一个
所以对于每个点必要删去一条边,删去的边相当于我们不要的选项 剩下的和S,T相连的边相当于我们的选择 此时删去的肯定是最小的边.
接下来我们要处理组合效应的问题.
每个组合效应有三种选择:A/B/无
这样对于每个组合只建一个点很难满足要求 则我们把每个组合拆成A,B两个点 A点和S建边(S,A,C1[i]) B点和T建边(B,T,C2[i]) 表示选择A,B能得到的贡献.
再对于组合里的每个数都连边(A,K[i],INF) (K[i],B,INF) 这样图中除边权为INF的边的边权减去跑出来的最大流即为答案.
为什么这样跑出来即是我们选择要删去的选项?
因为最小割不可能会割INF的边
每个组合效应的A点 他旗下的每个点要都选A他才能产生贡献,如果有一个选了B则会产生增广路径,那么就必须要割掉(S,A,C1[i])
每个组合效应的B点 他旗下的每个点要都选B他才能产生贡献,如果有一个选了A则同样会产生增广路径,必须要割掉(B,T,C2[i])
- #include <cstdio>
- #include <cstring>
- #include <queue>
- using namespace std;
- const int N=;
- const int M=;
- const int inf=0x3f3f3f3f;
- int head[N],edge[M],to[M],next[M],cnt=;
- void add(int u,int v,int w)
- {
- to[++cnt]=v;next[cnt]=head[u];edge[cnt]=w;head[u]=cnt;
- to[++cnt]=u;next[cnt]=head[v];edge[cnt]=;head[v]=cnt;
- }
- int dep[N],used[N],pre[N],tot,s[N],ans,m,n,sum;
- queue <int > q;
- bool bfs()
- {
- while(!q.empty()) q.pop();
- q.push();
- memset(dep,,sizeof(dep));
- dep[]=;
- while(!q.empty()&&q.front()!=n+)
- {
- int u=q.front();
- q.pop();
- for(int i=head[u];i;i=next[i])
- {
- int v=to[i],w=edge[i];
- if(!dep[v]&&w)
- {
- dep[v]=dep[u]+;
- q.push(v);
- }
- }
- }
- return !q.empty();
- }
- int main()
- {
- scanf("%d",&n);
- int w,v,k,c1,c2;
- for(int i=;i<=n;i++)
- {
- scanf("%d",&w);
- sum+=w;
- add(,i,w);
- }
- for(int i=;i<=n;i++)
- {
- scanf("%d",&w);
- sum+=w;
- add(i,n+,w);
- }
- scanf("%d",&m);
- for(int i=;i<=m;i++)
- {
- scanf("%d%d%d",&k,&c1,&c2);
- add(,i+n+,c1);sum+=c1;
- add(i+n+m+,n+,c2);sum+=c2;
- for(int j=;j<=k;j++)
- {
- scanf("%d",&v);
- add(i+n+,v,inf);
- add(v,i+n+m+,inf);
- }
- }
- while(bfs())
- {
- memset(used,,sizeof(used));
- s[++tot]=;
- while(tot)
- {
- int u=s[tot];
- if(u==n+)
- {
- int mi=inf,id;
- for(int i=tot;i>;i--)
- if(mi>=edge[pre[s[i]]])
- {
- mi=edge[pre[s[i]]];
- id=i;
- }
- ans+=mi;
- for(int i=tot;i>;i--)
- {
- edge[pre[s[i]]]-=mi;
- edge[pre[s[i]]^]+=mi;
- }
- tot=id-;
- used[n+]=;
- }
- else
- {
- for(int i=head[u];i;i=next[i])
- {
- int v=to[i],w=edge[i];
- if(!used[v]&&dep[v]==dep[u]+&&w)
- {
- used[v]=;
- s[++tot]=v;
- pre[v]=i;
- break;
- }
- }
- if(u==s[tot]) tot--;
- }
- }
- }
- printf("%d\n",sum-ans);
- return ;
- }
P1361 小M的作物 最小割理解的更多相关文章
- 洛谷 - P1361 - 小M的作物 - 最小割 - 最大权闭合子图
第一次做最小割,不是很理解. https://www.luogu.org/problemnew/show/P1361 要把东西分进两类里,好像可以应用最小割的模板,其中一类A作为源点,另一类B作为汇点 ...
- [P1361] 小M的作物 - 最小割
没想到今天早上的第一题网络流就血了这么多发 从经典的二选一问题上魔改 仍然考虑最小割 #include <bits/stdc++.h> using namespace std; #defi ...
- BZOJ 3438: 小M的作物( 最小割 )
orz出题人云神... 放上官方题解... 转成最小割然后建图跑最大流就行了... ---------------------------------------------------------- ...
- BZOJ3438小M的作物——最小割
题目描述 小M在MC里开辟了两块巨大的耕地A和B(你可以认为容量是无穷),现在,小P有n中作物的种子,每种作物的种子 有1个(就是可以种一棵作物)(用1...n编号),现在,第i种作物种植在A中种植可 ...
- 【BZOJ3438】小M的作物 最小割
[BZOJ3438]小M的作物 Description 小M在MC里开辟了两块巨大的耕地A和B(你可以认为容量是无穷),现在,小P有n中作物的种子,每种作物的种子 有1个(就是可以种一棵作物)(用1. ...
- 3438: 小M的作物[最小割]
3438: 小M的作物 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1073 Solved: 465[Submit][Status][Discus ...
- 【BZOJ-3438】小M的作物 最小割 + 最大权闭合图
3438: 小M的作物 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 825 Solved: 368[Submit][Status][Discuss ...
- 小M的作物 最小割最大流
题目描述 小M在MC里开辟了两块巨大的耕地A和B(你可以认为容量是无穷),现在,小P有n中作物的种子,每种作物的种子有1个(就是可以种一棵作物)(用1...n编号). 现在,第i种作物种植在A中种植可 ...
- 洛谷 P1361 小M的作物 解题报告
P1361 小M的作物 题目描述 小M在MC里开辟了两块巨大的耕地\(A\)和\(B\)(你可以认为容量是无穷),现在,小\(P\)有\(n\)中作物的种子,每种作物的种子有1个(就是可以种一棵作物) ...
随机推荐
- centos(linux)--vsftpd配置
1.安装 执行 yum -y install vsftpd 注:(1)是否使用sudo权限根据个人的具体情况 (2)rpm -qa | grep vsftpd 可以通过这个检查是否已经安装vsftpd ...
- 01.03 vim编辑器使用
==========linux基础命令的使用==========================绝对路径:由根目录(/)开始写起的文件名或目录名称相对路径:相对于目前路径的文件名写法(开头不是/就属于 ...
- 【DSP开发】6455EMIF
外部设备连接接口包括外部存储器连接接口(EMIF).主机接口(HPI)等.外部存储器接口主要用来同并行存储器连接,这些存储器包括SDRAM.SBSRAM.Flash.SRAM存储器等,外部存储器接口 ...
- 【SVN】总结:svn“Previous operation has not finished; run 'cleanup' if it was interrupted“
svn执行clean up命令时报错“Previous operation has not finished; run 'cleanup' if it was interrupted”.无论你到那个父 ...
- Gradle DSL Walle渠道包后安装启动APP
DSL(Domain-Specific Language) Gradle 是一个编译打包工具,但实际上它也是一个编程框架. Task 是 Gradle 中的一种数据类型,它代表了一些要执行或者要干的工 ...
- SQL SERVER YEAR函数
定义: YEAR函数返回指定日期的年的部分 语法: YEAR(date) 参数: ①date参数是合法的日期表达式. 返回值: int型数据 例: 声明:本文是本人查阅网上及书籍等各种资料,再加上自 ...
- mysql中比较实用的几个函数
1.曾有这样的需求: 可以使用如下函数: 语法:FIND_IN_SET(str,strlist). 定义: 1. 假如字符串str在由N子链组成的字符串列表strlist中,则返回值的范围在1到N之间 ...
- Dao设计模式简单实现
一.什么是Dao设计模式 Dao设计模式封装了操作具体数据库的细节,对业务层提供操作数据库的接口,因此降低了业务层代码与具体数据库之间的耦合,有利于人员分工,增加了程序的可移植性. Dao设计模式中主 ...
- 【hash】Seek the Name, Seek the Fame
[哈希和哈希表]Seek the Name, Seek the Fame 题目描述 The little cat is so famous, that many couples tramp over ...
- android中sqlite数据库的基本使用和添加多张表
看了很多关于android使用sqlite数据库的文章,很多都是介绍了数据库的建立和表的建立,而表通常都是只建立一张,而实际情况我们用到的表可能不止一张,那这种情况下我们又该怎么办呢,好了,下面我教大 ...