原题

给一个N*N的方阵,从[1,1]到[n,n]走K次,走过每个方格加上上面的数,然后这个格上面的数变为0。求可取得的最大的值。

要求最大值,所以把边权全为负跑最小费用即可。因为只有第一次经过该点的时候会得到价值,所以我们将一个点拆为两个,连一条容量为1费用为负权的边和一条容量为k-1费用为0的边。然后和右和下的点连边为容量为k,费用为0的边。跑费用流即可。

#include<cstdio>
#include<queue>
#include<algorithm>
#include<cstring>
#define N 5010
#define inf 0x3f3f3f3f
using namespace std;
int n,m,head[N],dis[N],cur[N],ans,cnt=2,s,t,ANS,T,a[55][55],k;
queue <int> q;
bool vis[N];
struct hhh
{
int to,next,w,cost;
}edge[N*N]; int read()
{
int ans=0,fu=1;
char j=getchar();
for (;(j<'0' || j>'9') && j!='-';j=getchar()) ;
if (j=='-') fu=-1,j=getchar();
for (;j>='0' && j<='9';j=getchar()) ans*=10,ans+=j-'0';
return ans*fu;
} void add(int u,int v,int w,int c)
{
edge[cnt].to=v;
edge[cnt].w=w;
edge[cnt].next=head[u];
edge[cnt].cost=c;
head[u]=cnt++;
} void addEdge(int u,int v,int w,int c)
{
add(u,v,w,c);
add(v,u,0,-c);
} bool bfs()
{
for (int i=s;i<=t;i++)
vis[i]=0,cur[i]=head[i],dis[i]=inf;
q.push(s);
dis[s]=0;
vis[s]=1;
while(!q.empty())
{
int r=q.front();
q.pop();
vis[r]=0;
for (int i=head[r],v;i;i=edge[i].next)
{
v=edge[i].to;
if (edge[i].w>0 && dis[r]+edge[i].cost<dis[v])
{
dis[v]=dis[r]+edge[i].cost;
if (!vis[v])
{
vis[v]=1;
q.push(v);
}
}
}
}
return dis[t]!=inf;
} int dfs(int x,int f)
{
if (x==t) return ANS+=f*dis[t],f;
int ha=0,now;
vis[x]=1;
for (int &i=cur[x],v;i;i=edge[i].next)
{
v=edge[i].to;
if (vis[v]) continue;
if (edge[i].w>0 && dis[v]==dis[x]+edge[i].cost)
{
now=dfs(v,min(f-ha,edge[i].w));
if (now)
{
ha+=now;
edge[i].w-=now;
edge[i^1].w+=now;
}
}
if (ha==f) return ha;
}
return ha;
} int main()
{
n=read();
k=read();
for (int i=1;i<=n;i++)
for (int j=1;j<=n;j++)
a[i][j]=read();
s=0;
t=2*n*n+1;
addEdge(s,1,k,0);
for (int i=1;i<=n;i++)
for (int j=1,p;j<=n;j++)
{
p=n*(i-1)+j;
addEdge(p,p+n*n,1,-a[i][j]);
addEdge(p,p+n*n,k-1,0);
if (i<n) addEdge(p+n*n,p+n,k,0);
if (j<n) addEdge(p+n*n,p+1,k,0);
}
addEdge(2*n*n,t,k,0);
while (bfs()) ans+=dfs(s,inf);
printf("%d",-ANS);
return 0;
}

最新文章

  1. [Window Title] (没有登录) [Content] ORA-12514: TNS: 监听程序当前无法识别连接描述符中请求的服务 [OK]
  2. C++多线程编程(入门实例)
  3. FZU2082树链剖分
  4. Ext中获取button的思考
  5. cocos2dx2.2.2登录场景中Checkbox选择框的实现
  6. 有关gcc的扩展__attribute__((unused))
  7. Java学习之IO字节流
  8. do{...}while(0)的妙用(转)
  9. jQuery鼠标移入移出(冒泡版和无冒泡版)
  10. 省市区地址三级联动jQuery插件Distpicker使用
  11. Oracle错误——tablespace &#39;XXXX&#39; does not exist
  12. kali linux wmtools安装
  13. 手撸orm框架
  14. STM32F4, USB HS with ULPI and Suspend/Wakeup
  15. 初学node.js-nodejs中实现HTTP服务(3)
  16. 【Python系列】Python包管理器pip
  17. python打开浏览器的三种方法
  18. openssl-0.9.8y
  19. 批处理文件——多个QQ一键登录
  20. Lambda表达式使用2

热门文章

  1. MySql错误1045 Access denied for user &#39;root&#39;@&#39;localhost&#39; (using password:YES)
  2. 经典sql语句汇总
  3. Linux终端显示控制字符
  4. webmin纯web界面管理linux系统
  5. 用ajax获取淘宝关键字接口
  6. python__系统 : socket_TCP相关
  7. php图片上传旋转压缩方法
  8. zeppelin的数据集的优化
  9. JS 金钱格式化
  10. 为什么i=i++后,i的值不变(深入解析)