#6122. 「网络流 24 题」航空路线问题

题目描述

给定一张航空图,图中顶点代表城市,边代表两个城市间的直通航线。现要求找出一条满足下述限制条件的且途经城市最多的旅行路线。

  1. 从最西端城市出发,单向从西向东途经若干城市到达最东端城市,然后再单向从东向西飞回起点(可途经若干城市)。
  2. 除起点城市外,任何城市只能访问一次。

对于给定的航空图,试设计一个算法找出一条满足要求的最佳航空旅行路线。

输入格式

第一行有两个正整数 NNN 和 VVV,NNN 表示城市数,VVV 表示直飞航线数。
接下来的 NNN 行中每一行是一个城市名,可乘飞机访问这些城市。城市名出现的顺序是从西向东。也就是说,设 i,ji,ji,j 是城市表列中城市出现的位置次序,当 i>ji>ji>j 时,表示 城市 iii在城市 jjj 的东边,而且不会有两个城市在同一条经线上。城市名是一个长度不超过 151515 的字符串,串中的字符可以是大小写字母或阿拉伯数字。例如,AGR34\text{AGR34}AGR34 或 BEL4\text{BEL4}BEL4。
再接下来的 VVV 行中,每行有两个城市名,中间用空格隔开,如 city1 city2\text{city1 city2}city1 city2 表示 city1\text{city1}city1 到 city2\text{city2}city2 有一条直通航线,从 city2\text{city2}city2 到 city1\text{city1}city1 也有一条直通航线。

输出格式

输出最佳航空旅行路线。
第一行是旅行路线中所访问的城市总数 MMM。
接下来的 M+1M+1M+1 行是旅行路线的城市名,每行一个。首先是出发城市名,然后按访问顺序列出其它城市名。注意,最后一行(终点城市)的城市名必然是出发城市名。如果有多组最优解,输出任意一组均可;如果问题无解,则输出 No Solution!

样例

样例输入

8 9
Vancouver
Yellowknife
Edmonton
Calgary
Winnipeg
Toronto
Montreal
Halifax
Vancouver Edmonton
Vancouver Calgary
Calgary Winnipeg
Winnipeg Toronto
Toronto Halifax
Montreal Halifax
Edmonton Montreal
Edmonton Yellowknife
Edmonton Calgary

样例输出

7
Vancouver
Edmonton
Montreal
Halifax
Toronto
Winnipeg
Calgary
Vancouver

数据范围与提示

对于所有数据,N<100N < 100N<100

#include<iostream>
#include<cstdio>
#include<cstring>
#include<map>
#include<queue>
#define maxn 110
using namespace std;
int dis[maxn],head[maxn],n,m,S,T,num=1,ans;
bool v[maxn],vis[maxn];
map<string,int>p;
struct node{int to,pre,v,w;}e[maxn*maxn];
struct Node{int e,v;}pre[maxn*maxn];
string pp[maxn],s1,s2;
void Insert(int from,int to,int v,int w){e[++num].to=to;e[num].v=v;e[num].w=w;e[num].pre=head[from];head[from]=num;e[++num].to=from;e[num].v=0;e[num].w=-w;e[num].pre=head[to];head[to]=num;
}
bool spfa(int x){memset(dis,0,sizeof(dis));memset(vis,0,sizeof(vis));queue<int>q;q.push(x);vis[x]=1;while(!q.empty()){int now=q.front();q.pop();vis[now]=0;for(int i=head[now];i;i=e[i].pre){int to=e[i].to;if(e[i].v>0&&dis[now]+e[i].w>dis[to]){dis[to]=dis[now]+e[i].w;pre[to].e=i;pre[to].v=now;if(!vis[to]){vis[to]=1;q.push(to);}}}}return dis[T];
}
int max_flow(int f){int res=0,d;while(f){if(!spfa(S))return -1;d=f;for(int i=T;i!=S;i=pre[i].v)d=min(e[pre[i].e].v,d);res+=d*dis[T];f-=d;for(int i=T;i!=S;i=pre[i].v){e[pre[i].e].v-=d;e[pre[i].e^1].v+=d;}}return res;
}
int main(){scanf("%d%d",&n,&m);S=1,T=n*2;for(int i=1;i<=n;i++){cin>>pp[i];p[pp[i]]=i;}for(int i=1;i<=m;i++){cin>>s1>>s2;int a1=p[s1],a2=p[s2];if(a1>a2)swap(a1,a2);if(a1==1&&a2==n)Insert(a1+n,a2,2,0);else Insert(a1+n,a2,1,0);}Insert(S,1+n,2,1);Insert(n,T,2,1);for(int i=2;i<n;i++)Insert(i,i+n,1,1);ans=max_flow(2);if(ans<0){puts("No Solution!");return 0;}printf("%d\n",ans-2);cout<<pp[1]<<endl;for(int i=head[S+n];i;i=e[i].pre)if(!e[i].v&&!(i&1)){int to=e[i].to;while(to){cout<<pp[to]<<endl;v[to]=1;int j;for(j=head[to+n],to=0;j;j=e[j].pre)if(!e[j].v&&!(j&1)){to=e[j].to;break;}}break;}for(int i=head[T-n];i;i=e[i].pre)if(!e[i^1].v&&(i&1)&&!v[e[i].to-n]){int to=e[i].to-n;while(to){cout<<pp[to]<<endl;v[to]=1;int j;for(j=head[to],to=0;j;j=e[j].pre)if(!e[j^1].v&&(j&1)){to=e[j].to-n;break;}}break;}
}

转载于:https://www.cnblogs.com/thmyl/p/8945431.html

loj #6122. 「网络流 24 题」航空路线问题相关推荐

  1. loj #6226. 「网络流 24 题」骑士共存问题

    #6226. 「网络流 24 题」骑士共存问题 题目描述 在一个 n×n\text{n} \times \text{n}n×n 个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示.棋盘上某些 ...

  2. loj #6004. 「网络流 24 题」圆桌聚餐(最大流)

    #6004. 「网络流 24 题」圆桌聚餐 内存限制:256 MiB时间限制:5000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统计讨论测试数 ...

  3. 【刷题】LOJ 6011 「网络流 24 题」运输问题

    题目描述 W 公司有 \(m\) 个仓库和 \(n\) 个零售商店.第 \(i\) 个仓库有 \(a_i\) 个单位的货物:第 \(j\) 个零售商店需要 \(b_j\) 个单位的货物.货物供需平衡, ...

  4. LOJ#6002. 「网络流 24 题」最小路径覆盖

    模板. 1 #include<iostream> 2 #include<cstring> 3 #include<cstdio> 4 //#include<ti ...

  5. 【刷题】LOJ 6014 「网络流 24 题」最长 k 可重区间集

    题目描述 给定实直线 \(L\) 上 \(n\) 个开区间组成的集合 \(I\) ,和一个正整数 \(k\) ,试设计一个算法,从开区间集合 \(I\) 中选取出开区间集合 \(S \subseteq ...

  6. 【刷题】LOJ 6005 「网络流 24 题」最长递增子序列

    题目描述 给定正整数序列 \(x_1 \sim x_n\) ,以下递增子序列均为非严格递增. 计算其最长递增子序列的长度 \(s\) . 计算从给定的序列中最多可取出多少个长度为 \(s\) 的递增子 ...

  7. 「网络流24题」 题目列表

    「网络流24题」 题目列表 序号 题目标题 模型 题解 1 飞行员配对方案问题 二分图最大匹配 <1> 2 太空飞行计划问题 最大权闭合子图 <2> 3 最小路径覆盖问题 二分 ...

  8. LibreOJ #6001. 「网络流 24 题」太空飞行计划 最大权闭合图

    #6001. 「网络流 24 题」太空飞行计划 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统计讨论测 ...

  9. LibreOJ 6004. 「网络流 24 题」圆桌聚餐 网络流版子题

    #6004. 「网络流 24 题」圆桌聚餐 内存限制:256 MiB时间限制:5000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统计讨论测试数 ...

  10. 「网络流24题」 12. 软件补丁问题

    「网络流24题」 12. 软件补丁问题 状压 DP,SPFA 转移. 没错,跟网络流没任何关系. b1.b2.f1.f2 都用二进制存下来,第 i 位表示是否有这个错误. 然后从每位都是 1 到 0 ...

最新文章

  1. C指针8:二级指针(意思就是指向指针的指针)
  2. DL之DilatedConvolutions:Dilated Convolutions(膨胀卷积/扩张卷积)算法的简介(论文介绍)、架构详解、案例应用等配图集合之详细攻略
  3. goroutine 那些事
  4. Regex 正则零宽断言
  5. Windows10避开注册的方式安装sourceTree,Failed to connect to github.com port 443: Timed out问题解决
  6. javadoc 开源项目_在下一个项目中不使用JavaDoc的5大原因
  7. 【转】WPF Expander 收缩不占空间的用法
  8. 简述 maven 命令 package、install、deploy 的区别
  9. java 判断客户端_javaweb服务端判断客户端使用的是手机还是电脑访问
  10. utilities(C++)——枚举
  11. windows 游戏编程大师技巧(第二版)学习之路-1
  12. 未受信任的企业级开发者_“未受信任的企业级开发者”是什么意思?怎么解决?...
  13. 我爱淘二次冲刺阶段2
  14. Android EditText 获得焦点不显示光标
  15. MacOS清理DNS缓存的终端代码推荐
  16. 热伤风和感冒有什么区别
  17. 一个网站做渗透测试对企业的安全性建议
  18. PAT A1010 Radix题解
  19. html 复选按钮 全选,JS实现复选按钮控件全选和批量操作
  20. 【20180615】【射频通信】RF、IF、Baseband的区别,PLL和VCO的定义,I/Q信号的定义

热门文章

  1. 游竹林寺不得,谈封山收费
  2. 原来喔趣是吾以前的同事开发的
  3. 电子设备的节能分析与设计
  4. vue插件挂载html6,vue6
  5. C#无法将顶级控件添加到控件 新的子窗体无法添加到主窗体
  6. Android的cangoback方法,Android应用开发Android8.0 WebView返回上一层失效(canGoBack返回false问题)解决办法...
  7. windows下怎么用basename_比较下养殖用铁丝网还是尼龙网,你会怎么选择?
  8. resultset需要关闭吗_你给家里的采暖壁挂炉做保养了吗?
  9. 缺少更新或刷新的键列信息_涨知识 | 刷新键究竟是用来干嘛的?你为什么要点击刷新呢?...
  10. Django Rest framework实现流程