题链:
http://www.lydsy.com/JudgeOnline/problem.php?id=3996
题解:
好题啊。
(不太熟悉矩阵相关,所以按某些博主的模型转换来理解的)
首先,那个式子可以化简为
D(某个数)=A * B * A' - C * A' ( A'为 A的倒置矩阵)
因为 A 为 01 矩阵,
把其考虑为 N个物品选或不选,
C[i]对应为i物品的花费,
而B[i,j]对应为同时选了i,j两个物品后带来的价值。
所以结合A,B,C的意义,用简单的矩阵知识去理解那个式子,
可以知道,D求得便是最大收益。
那么就转化为了 一个经典的最小割问题。(建图类似于网络流24道之太空飞行计划问题):
建立超源S,超汇T;
S -> (i,j) : B[i][j]
(i,j) -> (i) : INF
(i,j) -> (j) : INF
(i) -> T : C[i]
然后 ANS=sum(B)-最小割
代码:
#include<queue>
#include<cstdio>
#include<cstring>
#include<iostream>
#define MAXN 505*505
#define INF 0x3f3f3f3f
using namespace std;
struct Edge{
int to[MAXN*8],cap[MAXN*8],nxt[MAXN*8],head[MAXN*2],ent;
void Init(){
ent=2; memset(head,0,sizeof(head));
}
void Adde(int u,int v,int w){
to[ent]=v; cap[ent]=w; nxt[ent]=head[u]; head[u]=ent++;
to[ent]=u; cap[ent]=0; nxt[ent]=head[v]; head[v]=ent++;
}
int Next(int i,bool type){
return type?head[i]:nxt[i];
}
}E;
int cur[MAXN*2],d[MAXN*2];
int N,S,T,ANS;
int idx(int i,int j){
return j?(i-1)*N+j:N*N+i;
}
bool bfs(){
memset(d,0,sizeof(d));
queue<int>q; d[S]=1; q.push(S);
while(!q.empty()){
int u=q.front(); q.pop();
for(int i=E.Next(u,1);i;i=E.Next(i,0)){
int v=E.to[i];
if(d[v]||!E.cap[i]) continue;
d[v]=d[u]+1; q.push(v);
}
}
return d[T];
}
int dfs(int u,int reflow){
if(u==T||!reflow) return reflow;
int flowout=0,f;
for(int &i=cur[u];i;i=E.Next(i,0)){
int v=E.to[i];
if(!E.cap[i]||d[v]!=d[u]+1) continue;
f=dfs(v,min(reflow,E.cap[i]));
flowout+=f; E.cap[i^1]+=f;
reflow-=f; E.cap[i]-=f;
if(!reflow) break;
}
if(!flowout) d[u]=0;
return flowout;
}
int dinic(){//求最小割
int flow=0;
while(bfs()){
memcpy(cur,E.head,sizeof(E.head));
flow+=dfs(S,INF);
}
return flow;
}
int main()
{
E.Init();
scanf("%d",&N); S=N*N+N+1; T=S+1;
for(int i=1,x;i<=N;i++)
for(int j=1;j<=N;j++){
scanf("%d",&x); ANS+=x;
E.Adde(S,idx(i,j),x);
E.Adde(idx(i,j),idx(i,0),INF);
E.Adde(idx(i,j),idx(j,0),INF);
}
for(int i=1,x;i<=N;i++){
scanf("%d",&x);
E.Adde(idx(i,0),T,x);
}
ANS-=dinic();
printf("%d",ANS);
return 0;
}