C-Minimum grid
如果第iii行最大值是aaa,第jjj列最大值是ccc,我们需要第iii行单独有一个格子权值是aaa,而且第jjj列单独有一个格子权值是bbb,不过如果第iii行最大值和第jjj列最大值是都是aaa,那么我们只需让第iii行第jjj列的格子权值是aaa即可满足限制条件。显然我们用一个格子的代价肯定更小!
从大到小一次考虑每一种权值vvv,把权值最大值限制等于vvv的行放一边,而值最大值限制等于vvv的列放另一边,构成二分图。如果一个格子(i,j)(i,j)(i,j)需要填数,就在二分图中连一条从i→ji\to ji→j的边。
求出最大匹配我们就可以求出最多的限制用一个格子的代价。
不难发现每种权值互相不影响,因此建立多个二分图一次性求出所有二分图的最大匹配即可。
#include<bits/stdc++.h>
using namespace std;
using ll=long long;
template <class T=int> T rd()
{T res=0;T fg=1;char ch=getchar();while(!isdigit(ch)) {if(ch=='-') fg=-1;ch=getchar();}while( isdigit(ch)) res=(res<<1)+(res<<3)+(ch^48),ch=getchar();return res*fg;
}
const int N=2005;
int a[N],b[N];
int n,m,k;
vector<int> g[N];
int match[N];
bool vis[N];
ll ans;
bool dfs(int u)
{for(int v:g[u]){if(vis[v]) continue;vis[v]=1;if(!match[v]||dfs(match[v])){match[v]=u;return 1;}}return 0;
}
int main()
{n=rd(),m=rd(),k=rd();for(int i=1;i<=n;i++) a[i]=rd();for(int i=1;i<=n;i++) b[i]=rd();while(m--){int x=rd(),y=rd();if(a[x]==b[y]) g[x].push_back(y);}for(int i=1;i<=n;i++){memset(vis,0,sizeof vis);dfs(i);}for(int i=1;i<=n;i++) ans+=a[i]+b[i];for(int i=1;i<=n;i++)if(match[i]) ans-=b[i];printf("%lld\n",ans);
}