D - 文理分科 (网络流->最小割)

题目链接:https://cn.vjudge.net/contest/281959#problem/D

题目大意:中文题目

具体思路:我们需要求出最大的满意值,从另一方面想,我们可以求出总的满意值,然后再求出不符合情况的最小的代价,这两个相减,就能求出最大的满意值,这个时候就可以通过最小割来求了(最小割:使得整个图不连通的最小花费)。

这一篇博客讲的很好:https://blog.csdn.net/yakeding/article/details/79357545

AC代码:

  1 #include<iostream>
  2 #include<stack>
  3 #include<queue>
  4 #include<iomanip>
  5 #include<stdio.h>
  6 #include<cstring>
  7 #include<cstring>
  8 #include<cmath>
  9 #include<algorithm>
 10 #include<map>
 11 #include<vector>
 12 using namespace std;
 13 # define ll long long
 14 # define maxn 6000000+100
 15 # define inf 0x3f3f3f3f
 16 int prev[maxn];//边的编号
 17 int head[maxn];
 18 int f[2][4]= {{1,-1,0,0},{0,0,1,-1}};
 19 struct node
 20 {
 21     int to;
 22     int flow;
 23     int nex;
 24 } edge[maxn];
 25 int num,st,ed;
 26 void init()
 27 {
 28     memset(head,-1,sizeof(head));
 29     num=0;
 30 }
 31 void addedge(int fr,int to,int flow)
 32 {
 33     edge[num].to=to;
 34     edge[num].flow=flow;
 35     edge[num].nex=head[fr];
 36     head[fr]=num++;
 37     edge[num].to=fr;
 38     edge[num].flow=0;
 39     edge[num].nex=head[to];
 40     head[to]=num++;
 41 }
 42 bool bfs()
 43 {
 44     memset(prev,-1,sizeof(prev));
 45     prev[st]=1;
 46     queue<int>q;
 47     q.push(st);
 48     while(!q.empty())
 49     {
 50         int top=q.front();
 51         q.pop();
 52         for(int i=head[top]; i!=-1; i=edge[i].nex)
 53         {
 54             int temp=edge[i].to;
 55             if(prev[temp]==-1&&edge[i].flow>0)
 56             {
 57                 prev[temp]=prev[top]+1;
 58                 q.push(temp);
 59             }
 60         }
 61     }
 62     return prev[ed]!=-1;
 63 }
 64 int dfs(int u,int flow)
 65 {
 66     if(u==ed)
 67         return flow;
 68     int res=0;
 69     for(int i=head[u]; i!=-1; i=edge[i].nex)
 70     {
 71         int t=edge[i].to;
 72         if(prev[t]==(prev[u]+1)&&edge[i].flow>0)
 73         {
 74             int temp=dfs(t,min(flow,edge[i].flow));
 75             edge[i].flow-=temp;
 76             edge[i^1].flow+=temp;
 77             res+=temp;
 78             flow-=temp;
 79             if(flow==0)
 80                 break;
 81         }
 82     }
 83     if(res==0)
 84         prev[u]=-1;
 85     return res;
 86 }
 87 int n,m;
 88 int dinic()
 89 {
 90     int ans=0;
 91     while(bfs())
 92     {
 93         ans+=dfs(st,inf);
 94     }
 95     return ans;
 96 }
 97 bool judge(int t1,int t2)
 98 {
 99     if(t1>=1&&t1<=n&&t2>=1&&t2<=m)
100         return true;
101     return false;
102 }
103 int main()
104 {
105     init();
106     int sum=0;
107     int tmp;
108     st=1e5,ed=1e5+1;
109     scanf("%d %d",&n,&m);
110     for(int i=1; i<=n; i++)
111     {
112         for(int j=1; j<=m; j++)
113         {
114             scanf("%d",&tmp);
115             sum+=tmp;
116             addedge((i-1)*m+j,ed,tmp);
117         }
118     }
119     for(int i=1; i<=n; i++)
120     {
121         for(int j=1; j<=m; j++)
122         {
123             scanf("%d",&tmp);
124             sum+=tmp;
125             addedge(st,(i-1)*m+j,tmp);
126         }
127     }
128     for(int i=1; i<=n; i++)
129     {
130         for(int j=1; j<=m; j++)
131         {
132             scanf("%d",&tmp);
133             addedge((i-1)*m+j,((i-1)*m+j)+n*m,inf);
134             addedge(((i-1)*m+j)+n*m,ed,tmp);
135             sum+=tmp;
136             for(int k=0; k<4; k++)
137             {
138                 int x=i+f[0][k];
139                 int y=j+f[1][k];
140                 if(judge(x,y))
141                     addedge((x-1)*m+y,((i-1)*m+j)+n*m,inf);
142             }
143         }
144     }
145     for(int i=1; i<=n; i++)
146     {
147         for(int j=1; j<=m; j++)
148         {
149             scanf("%d",&tmp);
150             sum+=tmp;
151             addedge(((i-1)*m+j)+n*m*2,(i-1)*m+j,inf);
152             addedge(st,((i-1)*m+j)+n*m*2,tmp);
153             for(int k=0; k<4; k++)
154             {
155                 int x=i+f[0][k];
156                 int y=j+f[1][k];
157                 if(judge(x,y))
158                     addedge(((i-1)*m+j)+n*m*2,(x-1)*m+y,inf);
159             }
160         }
161     }
162    // cout<<1<<endl;
163     int ans=dinic();
164   //  cout<<1<<endl;
165     printf("%d\n",sum-ans);
166     return 0;
167 }

原文地址:https://www.cnblogs.com/letlifestop/p/10351293.html

时间: 2024-08-29 18:20:07

D - 文理分科 (网络流->最小割)的相关文章

【BZOJ3894】文理分科(最小割)

[BZOJ3894]文理分科(最小割) 题面 BZOJ Description 文理分科是一件很纠结的事情!(虽然看到这个题目的人肯定都没有纠 结过) 小P所在的班级要进行文理分科.他的班级可以用一个n*m的矩阵进行 描述,每个格子代表一个同学的座位.每位同学必须从文科和理科中选择 一科.同学们在选择科目的时候会获得一个满意值.满意值按如下的方式 得到: 1.如果第i行第秒J的同学选择了文科,则他将获得art[i][j]的满意值,如 果选择理科,将得到science[i][j]的满意值. 2.如

BZOJ_3894_文理分科&amp;&amp;BZOJ_2127_happiness_最小割

BZOJ_3894_文理分科_最小割 Description 文理分科是一件很纠结的事情!(虽然看到这个题目的人肯定都没有纠 结过) 小P所在的班级要进行文理分科.他的班级可以用一个n*m的矩阵进行 描述,每个格子代表一个同学的座位.每位同学必须从文科和理科中选择 一科.同学们在选择科目的时候会获得一个满意值.满意值按如下的方式 得到: 1.如果第i行第秒J的同学选择了文科,则他将获得art[i][j]的满意值,如 果选择理科,将得到science[i][j]的满意值. 2.如果第i行第J列的同

[BZOJ 3894] 文理分科 【最小割】

题目链接:BZOJ - 3894 题目分析 最小割模型,设定一个点与 S 相连表示选文,与 T 相连表示选理. 那么首先要加上所有可能获得的权值,然后减去最小割,即不能获得的权值. 那么对于每个点,从 S 向它连权值为它选文的价值的边,从它向 T 连权值为它选理的价值的边. 对于一个点,它和与它相邻的点构成了一个集合,这个集合如果都选文,可以获得一个价值v1,如果都选理,可以获得一个价值 v2. 只要这个集合中有一个点选文,就无法获得 v2,只要有一个点选理,就无法获得 v1. 那么处理方式就是

【bzoj3144】[Hnoi2013]切糕 网络流最小割

题目描述 输入 第一行是三个正整数P,Q,R,表示切糕的长P. 宽Q.高R.第二行有一个非负整数D,表示光滑性要求.接下来是R个P行Q列的矩阵,第z个 矩阵的第x行第y列是v(x,y,z) (1≤x≤P, 1≤y≤Q, 1≤z≤R). 100%的数据满足P,Q,R≤40,0≤D≤R,且给出的所有的不和谐值不超过1000. 输出 仅包含一个整数,表示在合法基础上最小的总不和谐值. 样例输入 2 2 2 1 6 1 6 1 2 6 2 6 样例输出 6 题目大意 给定一个p行q列的矩阵,每个位置可以

二分图&amp;网络流&amp;最小割等问题的总结

二分图基础: 最大匹配:匈牙利算法 最小点覆盖=最大匹配 最小边覆盖=总节点数-最大匹配 最大独立集=点数-最大匹配 网络流: 带下界网络流 最小割问题的总结: *意义 1.加inf的边表示不能被割,通常用于体现某个点必须属于某个集合 连边(s,u,w)代表如果u不在s割的话需要付出代价w 2.连边(u,v,w)代表如果u在s割,v在t割需要付出代价w 但注意,如果u在t割,v在s割是不需要付出代价的. 那么如果连边(u,v,w)以及(v,u,w)则说明当u与v所属割不同的时候需要付出代价w *

HDU 2435 There is a war (网络流-最小割)

There is a war Problem Description There is a sea. There are N islands in the sea. There are some directional bridges connecting these islands. There is a country called Country One located in Island 1. There is another country called Country Another

【bzoj3630】[JLOI2014]镜面通道 对偶图+计算几何+网络流最小割

题目描述 在一个二维平面上,有一个镜面通道,由镜面AC,BD组成,AC,BD长度相等,且都平行于x轴,B位于(0,0).通道中有n个外表面为镜面的光学元件,光学元件α为圆形,光学元件β为矩形(这些元件可以与其他元件和通道有交集,具体看下图).光线可以在AB上任一点以任意角度射入通道,光线不会发生削弱.当出现元件与元件,元件和通道刚好接触的情况视为光线无法透过(比如两圆相切).现在给出通道中所有元件的信息(α元件包括圆心坐标和半径xi,yi,ri,β元件包括左下角和右上角坐标x1,y1,x2,y2

【bzoj2127】happiness 网络流最小割

题目描述 高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友.这学期要分文理科了,每个同学对于选择文科与理科有着自己的喜悦值,而一对好朋友如果能同时选文科或者理科,那么他们又将收获一些喜悦值.作为计算机竞赛教练的scp大老板,想知道如何分配可以使得全班的喜悦值总和最大. 输入 第一行两个正整数n,m.接下来是六个矩阵第一个矩阵为n行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择文科获得的喜悦值.第二个矩阵为n行m列 此矩阵的第i行

【bzoj2132】圈地计划 网络流最小割

题目描述 最近房地产商GDOI(Group of Dumbbells Or Idiots)从NOI(Nuts Old Idiots)手中得到了一块开发土地.据了解,这块土地是一块矩形的区域,可以纵横划分为N×M块小区域.GDOI要求将这些区域分为商业区和工业区来开发.根据不同的地形环境,每块小区域建造商业区和工业区能取得不同的经济价值.更具体点,对于第i行第j列的区域,建造商业区将得到Aij收益,建造工业区将得到Bij收益.另外不同的区域连在一起可以得到额外的收益,即如果区域(I,j)相邻(相邻