zoukankan      html  css  js  c++  java
  • [洛谷P2045]方格取数加强版

    题目大意:有一个n*n的矩阵,每个格子有一个非负整数,规定一个人从(1,1)开始,只能往右或下走,走到(n,n)为止,并把沿途的数取走,取走后数变为0。这个人共取n次,求取得的数的最大总和。

    解题思路:由于取多少次不确定,所以不能用dp。

    我们发现,一个格子只能从左边或上面走来,且数只能取到一次,那么我们可以把此题转化为最大费用最大流问题。首先拆点,将一个点拆成x和y,然后从x到y连一条容量为1,流量为x(x为这格的数)的边,然后再连一条容量为inf,费用为0的边,这样即可保证一个点可以走多次,而数只能取一次。然后连接a和b时,从a的y向b的x连一条容量为inf,费用为0的边。最后跑最大费用最大流即可。

    实现时对于(i,j),我们把这个点的x编号为$(i-1)*n+j$,y编号为$(i-1)*n+j+n^2$即可。

    以下为EK算法代码。

    C++ Code:

    #include<cstdio>
    #include<vector>
    #include<queue>
    #include<string.h>
    using namespace std;
    #define inf 0x3f3f3f3f
    #define N 1000000
    struct edge{
    	int from,to,cap,cost,nxt;
    }e[N];
    int n,k,dis[N],a[N],pree[N],head[N],cnt;
    bool vis[N];
    queue<int>q;
    inline void addedge(int from,int to,int cap,int cost){
    	e[++cnt]=(edge){from,to,cap,cost,head[from]};
    	head[from]=cnt;
    	e[++cnt]=(edge){to,from,0,-cost,head[to]};
    	head[to]=cnt;
    }
    bool spfa(int s,int t,int& flow,int& cost){
    	memset(pree,0,sizeof(pree));
    	memset(a,0x3f,sizeof(a));
    	memset(dis,200,sizeof(dis));
    	memset(vis,0,sizeof(vis));
    	vis[s]=1;
    	dis[s]=0;
    	q.push(s);
    	while(!q.empty()){
    		int u=q.front();
    		q.pop();
    		vis[u]=0;
    		for(int i=head[u];i;i=e[i].nxt){
    			if(e[i].cap>0&&dis[e[i].to]<dis[u]+e[i].cost){
    				dis[e[i].to]=dis[u]+e[i].cost;
    				pree[e[i].to]=i;
    				if(a[u]>e[i].cap)a[e[i].to]=e[i].cap;else a[e[i].to]=a[u];
    				if(!vis[e[i].to]){
    					vis[e[i].to]=1;
    					q.push(e[i].to);
    				}
    			}
    		}
    	}
    	if(dis[t]<1)return false;
    	flow+=a[t];
    	cost+=a[t]*dis[t];
    	for(int i=t;i!=s;i=e[pree[i]].from){
    		e[pree[i]].cap-=a[t];
    		e[pree[i]^1].cap+=a[t];
    	}
    	return true;
    }
    int main(){
    	cnt=1;
    	scanf("%d%d",&n,&k);
    	int m=n*n;
    	for(int i=1;i<=n;++i){
    		for(int j=1;j<=n;++j){
    			int x;
    			scanf("%d",&x);
    			addedge((i-1)*n+j,(i-1)*n+j+m,inf,0);
    			addedge((i-1)*n+j,(i-1)*n+j+m,1,x);
    			if(i>1){
    				addedge((i-2)*n+j+m,(i-1)*n+j,inf,0);
    			}
    			if(j>1){
    				addedge((i-1)*n+j-1+m,(i-1)*n+j,inf,0);
    			}
    			if(i==1&&j==1){
    				addedge(0,1,inf,0);
    			}
    		}
    	}
    	int flow=0,cost=0;
    	while(k--)
    	if(!spfa(0,m<<1,flow,cost))break;
    	printf("%d
    ",cost);
    	return 0;
    }
    
  • 相关阅读:
    找到了2年前的一个微博小号
    Float Equal Problem
    有用的护肤品贴
    最近状态总结
    [Coursera]Machine Learning
    KMP算法(转载)
    [Leetcode] Median of Two Sorted Arrays
    [Algorithms(Princeton)] Week1
    [Algorithms(Princeton)] Week1
    [Leetcode] Binary Tree Maximum Path Sum
  • 原文地址:https://www.cnblogs.com/Mrsrz/p/7399814.html
Copyright © 2011-2022 走看看