loj #6122. 「网络流 24 题」航空路线问题
#6122. 「网络流 24 题」航空路线问题
题目描述
给定一张航空图,图中顶点代表城市,边代表两个城市间的直通航线。现要求找出一条满足下述限制条件的且途经城市最多的旅行路线。
- 从最西端城市出发,单向从西向东途经若干城市到达最东端城市,然后再单向从东向西飞回起点(可途经若干城市)。
- 除起点城市外,任何城市只能访问一次。
对于给定的航空图,试设计一个算法找出一条满足要求的最佳航空旅行路线。
输入格式
第一行有两个正整数 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=,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=;e[num].w=-w;e[num].pre=head[to];head[to]=num;
}
bool spfa(int x){
memset(dis,,sizeof(dis));
memset(vis,,sizeof(vis));
queue<int>q;
q.push(x);vis[x]=;
while(!q.empty()){
int now=q.front();q.pop();vis[now]=;
for(int i=head[now];i;i=e[i].pre){
int to=e[i].to;
if(e[i].v>&&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]=;q.push(to);}
}
}
}
return dis[T];
}
int max_flow(int f){
int res=,d;
while(f){
if(!spfa(S))return -;
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^].v+=d;
}
}
return res;
}
int main(){
scanf("%d%d",&n,&m);
S=,T=n*;
for(int i=;i<=n;i++){
cin>>pp[i];
p[pp[i]]=i;
}
for(int i=;i<=m;i++){
cin>>s1>>s2;
int a1=p[s1],a2=p[s2];
if(a1>a2)swap(a1,a2);
if(a1==&&a2==n)Insert(a1+n,a2,,);
else Insert(a1+n,a2,,);
}
Insert(S,+n,,);
Insert(n,T,,);
for(int i=;i<n;i++)Insert(i,i+n,,);
ans=max_flow();
if(ans<){
puts("No Solution!");
return ;
}
printf("%d\n",ans-);
cout<<pp[]<<endl;
for(int i=head[S+n];i;i=e[i].pre)
if(!e[i].v&&!(i&)){
int to=e[i].to;
while(to){
cout<<pp[to]<<endl;
v[to]=;
int j;
for(j=head[to+n],to=;j;j=e[j].pre)
if(!e[j].v&&!(j&)){
to=e[j].to;break;
}
}
break;
}
for(int i=head[T-n];i;i=e[i].pre)
if(!e[i^].v&&(i&)&&!v[e[i].to-n]){
int to=e[i].to-n;
while(to){
cout<<pp[to]<<endl;
v[to]=;
int j;
for(j=head[to],to=;j;j=e[j].pre)
if(!e[j^].v&&(j&)){
to=e[j].to-n;break;
}
}
break;
}
}
loj #6122. 「网络流 24 题」航空路线问题的更多相关文章
- 【刷题】LOJ 6122 「网络流 24 题」航空路线问题
题目描述 给定一张航空图,图中顶点代表城市,边代表两个城市间的直通航线.现要求找出一条满足下述限制条件的且途经城市最多的旅行路线. 从最西端城市出发,单向从西向东途经若干城市到达最东端城市,然后再单向 ...
- 【刷题】LOJ 6227 「网络流 24 题」最长k可重线段集问题
题目描述 给定平面 \(\text{xoy}\) 上 \(n\) 个开线段组成的集合 \(\text{I}\) ,和一个正整数 \(k\) ,试设计一个算法. 从开线段集合 \(\text{I}\) ...
- [luogu_P1251][LOJ#6008]「网络流 24 题」餐巾计划
[luogu_P1251][LOJ#6008]「网络流 24 题」餐巾计划 试题描述 一个餐厅在相继的 \(N\) 天里,第 \(i\) 天需要 \(R_i\) 块餐巾 \((i=l,2,-,N)\) ...
- [LOJ#6002]「网络流 24 题」最小路径覆盖
[LOJ#6002]「网络流 24 题」最小路径覆盖 试题描述 给定有向图 G=(V,E).设 P 是 G 的一个简单路(顶点不相交)的集合.如果 V 中每个顶点恰好在 P 的一条路上,则称 P 是 ...
- loj #6014. 「网络流 24 题」最长 k 可重区间集
#6014. 「网络流 24 题」最长 k 可重区间集 题目描述 给定实直线 L LL 上 n nn 个开区间组成的集合 I II,和一个正整数 k kk,试设计一个算法,从开区间集合 I II 中选 ...
- loj #6013. 「网络流 24 题」负载平衡
#6013. 「网络流 24 题」负载平衡 题目描述 G 公司有 n nn 个沿铁路运输线环形排列的仓库,每个仓库存储的货物数量不等.如何用最少搬运量可以使 n nn 个仓库的库存数量相同.搬运货物时 ...
- loj #6121. 「网络流 24 题」孤岛营救问题
#6121. 「网络流 24 题」孤岛营救问题 题目描述 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂, ...
- loj #6226. 「网络流 24 题」骑士共存问题
#6226. 「网络流 24 题」骑士共存问题 题目描述 在一个 n×n\text{n} \times \text{n}n×n 个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示.棋盘上 ...
- [loj #6003]「网络流 24 题」魔术球 二分图最小路径覆盖,网络流
#6003. 「网络流 24 题」魔术球 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统计讨论测试数据 ...
随机推荐
- leetcode481
public class Solution { public int MagicalString(int n) { ) ; ) ; ]; a[] = ; a[] = ; a[] = ; , tail ...
- 【284】◀▶ arcpy.da & arcpy 数据访问模块
使用游标访问数据 数据访问模块 (arcpy.da) 参考: ArcGIS Python编程案例(9)-ArcPy数据访问模块 读取几何 写入几何 使用 Python 指定查询 01 da.Sea ...
- Qt5信号和槽机制
信号槽是 Qt 框架引以为豪的机制之一.熟练使用和理解信号槽,能够设计出解耦的非常漂亮的程序,有利于增强我们的技术设计能力. 所谓信号槽,实际就是观察者模式.当某个事件发生之后,比如,按钮检测到自己被 ...
- jdbc中Statement和PreparedStatement有什么区别?哪个性能更好?
Statement和PreparedStatement的功能主要是对sql语句的执行 区别 (1)Statement每执行一条sql语句就需要生成一条执行计划,执行100条就需要100条执行计划Pre ...
- ensemble 的2篇入门 文章
python 篇: http://machinelearningmastery.com/ensemble-machine-learning-algorithms-python-scikit-learn ...
- 【原创】ListView快速滚动至新添加一行(自动滚动)
在C#开发中我们经常要开发一些日志系统,尤其是基于ListView的日志显示系统.但是当日志增多是你是否有一些困扰,就是它为什么不会自动滚动至最后一行. 以下是一小段代码,希望可以帮助你. publi ...
- 如何偷Android的内存-Tricking Android MemoryFile
之前在做一个内存优化的时候,使用到了MemoryFile,由此发现了MemoryFile的一些特性以及一个非常trickly的使用方法,因此在这里记录一下 What is it MemoryFile是 ...
- TinyMCE4.x整合教程-Xproer.WordPaster
版权所有 2009-2017 荆门泽优软件有限公司 保留所有权利 官方网站:http://www.ncmem.com/ 产品首页:http://www.ncmem.com/webplug/wordpa ...
- Discrete cosine transform(离散余弦转换)
A discrete cosine transform (DCT) expresses a finite sequence of data points in terms of a sum of co ...
- 编写高质量代码改善C#程序的157个建议——建议11: 区别对待==和Equals
建议11: 区别对待==和Equals 在开始本建议之前,首先要明确概念“相等性”.CLR中将“相等性”分为两类:“值相等性”和“引用相等性”.如果用来比较的两个变量所包含的数值相等,那么将其定义为“ ...