POJ-2135 Farm Tour---最小费用最大流模板题(构图)
题目链接:
https://vjudge.net/problem/POJ-2135
题目大意:
主人公要从1号走到第N号点,再重N号点走回1号点,同时每条路只能走一次。
这是一个无向图。输入数据第一行是2个是N和M。N为点的数量,M为路径个数。
接下来M行是边的数据,每行输入3个数,边的两个端点a,b和边的长度v。
要你输出来回最短的路径长度。
题目确保存在来回的不重复路径
解题思路:
这题可以转换成网络流的费用流。
来回并且路径不相同就相当于有用两条从1到N的路径。
把路径长度当成网络流里面每个流的费用,流量都设置成1这样就代表每条路径只能使用1次。增加2个点,源点和汇点,因为来回,就把源点到1建立一条流,流量为2(来回)费用为0,同样N到汇点建立一条流,流量为2费用为0。(保证只有两条路从源点到汇点,就是答案的解)这样一个网络流就出来了。
这里输入一条边要建4条边,首先建a->b的有向边,要同时建立反向边,再建b->a的有向边,一样建立反向边。
其他的就是模板了
注意:有可能会有这样一个问题,一条无向边拆分成两条有向边,有没有可能会把这两条有向边都走了呢,答案是否定的,因为求的是最小费用,如果一条边正向反向均走了一次,那么总流量为0,而且还有额外的费用,而我们算法的策略是每次都取最短路(也就是最小费用)找增广路,所以不可能找出费用为正数流量为0的情况,所以放心的敲模板吧。
- #include<iostream>
- #include<vector>
- #include<cstring>
- #include<cstdio>
- #include<queue>
- using namespace std;
- const int INF = 0x3f3f3f3f;
- const int maxn = + ;
- struct edge
- {
- int u, v, c, f, cost;
- edge(int u, int v, int c, int f, int cost):u(u), v(v), c(c), f(f), cost(cost){}
- };
- vector<edge>e;
- vector<int>G[maxn];
- int a[maxn];//找增广路每个点的水流量
- int p[maxn];//每次找增广路反向记录路径
- int d[maxn];//SPFA算法的最短路
- int inq[maxn];//SPFA算法是否在队列中
- int n, m;
- void init(int n)
- {
- for(int i = ; i <= n; i++)G[i].clear();
- e.clear();
- }
- void addedge(int u, int v, int c, int cost)
- {
- e.push_back(edge(u, v, c, , cost));
- e.push_back(edge(v, u, , , -cost));
- int m = e.size();
- G[u].push_back(m - );
- G[v].push_back(m - );
- }
- bool bellman(int s, int t, int& flow, long long & cost)
- {
- for(int i = ; i <= n + ; i++)d[i] = INF;//Bellman算法的初始化
- memset(inq, , sizeof(inq));
- d[s] = ;inq[s] = ;//源点s的距离设为0,标记入队
- p[s] = ;a[s] = INF;//源点流量为INF(和之前的最大流算法是一样的)
- queue<int>q;//Bellman算法和增广路算法同步进行,沿着最短路拓展增广路,得出的解一定是最小费用最大流
- q.push(s);
- while(!q.empty())
- {
- int u = q.front();
- q.pop();
- inq[u] = ;//入队列标记删除
- for(int i = ; i < G[u].size(); i++)
- {
- edge & now = e[G[u][i]];
- int v = now.v;
- if(now.c > now.f && d[v] > d[u] + now.cost)
- //now.c > now.f表示这条路还未流满(和最大流一样)
- //d[v] > d[u] + e.cost Bellman 算法中边的松弛
- {
- d[v] = d[u] + now.cost;//Bellman 算法边的松弛
- p[v] = G[u][i];//反向记录边的编号
- a[v] = min(a[u], now.c - now.f);//到达v点的水量取决于边剩余的容量和u点的水量
- if(!inq[v]){q.push(v);inq[v] = ;}//Bellman 算法入队
- }
- }
- }
- if(d[t] == INF)return false;//找不到增广路
- flow += a[t];//最大流的值,此函数引用flow这个值,最后可以直接求出flow
- cost += (long long)d[t] * (long long)a[t];//距离乘上到达汇点的流量就是费用
- for(int u = t; u != s; u = e[p[u]].u)//逆向存边
- {
- e[p[u]].f += a[t];//正向边加上流量
- e[p[u] ^ ].f -= a[t];//反向边减去流量 (和增广路算法一样)
- }
- return true;
- }
- int MincostMaxflow(int s, int t, long long & cost)
- {
- cost = ;
- int flow = ;
- while(bellman(s, t, flow, cost));//由于Bellman函数用的是引用,所以只要一直调用就可以求出flow和cost
- return flow;//返回最大流,cost引用可以直接返回最小费用
- }
- int main()
- {
- cin >> n >> m;
- int u, v, c;
- for(int i = ; i < m; i++)
- {
- cin >> u >> v >> c;
- addedge(u, v, , c);
- addedge(v, u, , c);
- }
- int s = , t = n + ;
- addedge(s, , , );//超级源点,边可通过两次,所以流量设成2,费用为0
- addedge(n, t, , );//超级汇点,边可通过两次,流量设成2,费用为0
- long long ans;
- MincostMaxflow(s, t, ans);
- cout<<ans<<endl;
- return ;
- }
POJ-2135 Farm Tour---最小费用最大流模板题(构图)的更多相关文章
- poj 2135 Farm Tour 最小费用最大流建图跑最短路
题目链接 题意:无向图有N(N <= 1000)个节点,M(M <= 10000)条边:从节点1走到节点N再从N走回来,图中不能走同一条边,且图中可能出现重边,问最短距离之和为多少? 思路 ...
- POJ 2135 Farm Tour [最小费用最大流]
题意: 有n个点和m条边,让你从1出发到n再从n回到1,不要求所有点都要经过,但是每条边只能走一次.边是无向边. 问最短的行走距离多少. 一开始看这题还没搞费用流,后来搞了搞再回来看,想了想建图不是很 ...
- poj 2351 Farm Tour (最小费用最大流)
Farm Tour Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 17230 Accepted: 6647 Descri ...
- Farm Tour(最小费用最大流模板)
Farm Tour Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 18150 Accepted: 7023 Descri ...
- [poj] 1235 Farm Tour || 最小费用最大流
原题 费用流板子题. 费用流与最大流的区别就是把bfs改为spfa,dfs时把按deep搜索改成按最短路搜索即可 #include<cstdio> #include<queue> ...
- POJ2135 Farm Tour —— 最小费用最大流
题目链接:http://poj.org/problem?id=2135 Farm Tour Time Limit: 1000MS Memory Limit: 65536K Total Submis ...
- TZOJ 1513 Farm Tour(最小费用最大流)
描述 When FJ's friends visit him on the farm, he likes to show them around. His farm comprises N (1 &l ...
- POJ2135 最小费用最大流模板题
练练最小费用最大流 此外此题也是一经典图论题 题意:找出两条从s到t的不同的路径,距离最短. 要注意:这里是无向边,要变成两条有向边 #include <cstdio> #include ...
- 【网络流#2】hdu 1533 - 最小费用最大流模板题
最小费用最大流,即MCMF(Minimum Cost Maximum Flow)问题 嗯~第一次写费用流题... 这道就是费用流的模板题,找不到更裸的题了 建图:每个m(Man)作为源点,每个H(Ho ...
- 2018牛客网暑期ACM多校训练营(第五场) E - room - [最小费用最大流模板题]
题目链接:https://www.nowcoder.com/acm/contest/143/E 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K ...
随机推荐
- PAT1089【归并排序】
这题略...恶心.. 他说归并排序依次是相邻有序两块合并,而一向打惯了递归??? #include <bits/stdc++.h> using namespace std; typedef ...
- 用vue写一个仿简书的轮播图
原文地址:用vue写一个仿简书的轮播图 先展示最终效果: Vue的理念是以数据驱动视图,所以拒绝通过改变元素的margin-top来实现滚动效果.写好css样式,只需改变每张图片的class即可实现轮 ...
- 在pom包中添加spring-boot-starter-test包引用
有很多网友会时不时的问我,spring boot项目如何测试,如何部署,在生产中有什么好的部署方案吗?这篇文章就来介绍一下spring boot 如何开发.调试.打包到最后的投产上线. 开发阶段 单元 ...
- C#正则表达式快速入门
作者将自己在学习正则表达式中的心得和笔记作了个总结性文章,希望对初学C#正则表达式的读者有帮助. [内容] 什么是正则表达式 涉及的基本的类 正则表达式基础知识 构建表达式基本方法 编写一个检验程序 ...
- UVA - 12563 Jin Ge Jin Qu hao (01背包)
InputThe first line contains the number of test cases T (T ≤ 100). Each test case begins with two po ...
- MarkDown基础语法大全
一.MarkDown是什么? Markdown是一种轻量级的「标记语言」,创始人为约翰·格鲁伯,用简洁的语法代替排版,目前被越来越多的知识工作者.写作爱好者.程序员或研究员广泛使用.其常用的标记符号不 ...
- [Leetcode]007. Reverse Integer
public class Solution { public int reverse(int x) { long rev=0; while(x!=0){ rev = rev*10+x%10; x=x/ ...
- 华东交通大学2017年ACM“双基”程序设计竞赛 1001
Problem Description 最近流行吃鸡,那就直接输出一行"Winner winner ,chicken dinner!"(没有双引号)模板代码:#include &l ...
- .NET Memcached Client 扩展获取所有缓存Key
.NET Memcached Client默认实现中并没有获取所有已经缓存Key的方法,但在业务中有时候需求中需要通过正则删除符合条件的缓存内容,所以就要通过读取已经缓存Key进行相关的匹配,然后删除 ...
- 11-----broder(边框)
边框 border:边框的意思,描述盒子的边框 边框有三个要素: 粗细 线性样式 颜色 如果颜色不写,默认是黑色.如果粗细不写,不显示边框.如果只写线性样式,默认的有上下左右 3px的宽度,实体样式, ...