\(\color{#0066ff}{题目描述}\)

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

(1)从最西端城市出发,单向从西向东途经若干城市到达最东端城市,然后再单向从东向西飞回起点(可途经若干城市)。

(2)除起点城市外,任何城市只能访问 1 次。

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

\(\color{#0066ff}{输入格式}\)

第 1 行有 2 个正整数 N 和 V,N 表示城市数,N<100,V 表示直飞航线数。

接下来的 N 行中每一行是一个城市名,可乘飞机访问这些城市。城市名出现的顺序是从西向东。也就是说,设 i,j 是城市表列中城市出现的顺序,当 i>j 时,表示城市 i 在城市 j 的东边,而且不会有 2 个城市在同一条经线上。城市名是一个长度不超过15 的字符串,串中的字符可以是字母或阿拉伯数字。例如,AGR34 或 BEL4。

再接下来的 V 行中,每行有 2 个城市名,中间用空格隔开,如 city1 city2 表示 city1到 city2 有一条直通航线,从 city2 到 city1 也有一条直通航线。

\(\color{#0066ff}{输出格式}\)

文件第 1 行是旅行路线中所访问的城市总数 M。 接下来的 M+1 行是旅行路线的城市名,每行写 1 个城市名。首先是出发城市名,然后按访问顺序列出其它城市名。 注意,最后 1 行(终点城市)的城市名必然是出发城市名。如果问题无解,则输出“No Solution!”。

\(\color{#0066ff}{输入样例}\)

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

\(\color{#0066ff}{输出样例}\)

7
Vancouver
Edmonton
Montreal
Halifax
Toronto
Winnipeg
Calgary
Vancouver

\(\color{#0066ff}{数据范围与提示}\)

有spj

\(\color{#0066ff}{题解}\)

题意就是求两条不相交的从1-n的路径,使得经过的点尽可能多

所以,若本图合法,则跑完最大流一定\(\leq 2\)

跑完就知道走了哪些点

因为要使路径最长

所以跑费用流,对边赋权为1跑最长路

每个点要拆点,拆成的两个点连权为1的边

s,t作为起点终点,再练一条权为0的边

对于图中所给边,直接连权为0的边就行

最后重新建图,dfs

#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<map>
#include<queue>
#include<string>
#define _ 0
#define LL long long
inline LL in()
{
LL x=0,f=1; char ch;
while(!isdigit(ch=getchar()))(ch=='-')&&(f=-f);
while(isdigit(ch)) x=x*10+(ch^48),ch=getchar();
return x*f;
}
const int inf=0x7fffffff;
int n,v,s,t,cnt=1;
std::queue<int> q;
struct node
{
int to,dis,nxt,can;
node(int to=0,int dis=0,int nxt=0,int can=0):to(to),dis(dis),nxt(nxt),can(can){}
}e[1050500];
int dis[1200],change[1200],head[1200],road[1200];
bool vis[1200],flag;
std::map<std::string,int> mp;
std::string f[1200],ss1,ss2;
int aa[1200],bb[1200];
int h[1200],st[1200],top;
inline void add(int from,int to,int dis,int can)
{
e[++cnt]=node(to,dis,head[from],can);
head[from]=cnt;
}
inline void link(int from,int to,int dis,int can)
{
add(from,to,dis,can);
add(to,from,-dis,0);
}
inline bool spfa()
{
for(int i=1;i<=n<<1;i++) dis[i]=-inf,change[i]=inf;
dis[s]=0;
q.push(s);
while(!q.empty())
{
int tp=q.front(); q.pop();
vis[tp]=false;
for(int i=head[tp];i;i=e[i].nxt)
{
int go=e[i].to;
if(dis[go]<dis[tp]+e[i].dis&&e[i].can>0)
{
dis[go]=dis[tp]+e[i].dis;
change[go]=std::min(change[tp],e[i].can);
road[go]=i;
if(!vis[go]) vis[go]=true,q.push(go);
}
}
}
return change[t]!=inf;
}
inline void add(int from,int to)
{
e[++cnt]=node(to,0,h[from],0);
h[from]=cnt;
}
inline void dfs(int x)
{
st[++top]=x;
vis[x]=1;
for(int i=h[x];i;i=e[i].nxt)
{
int go=e[i].to;
if(!vis[go]) dfs(go);
}
}
inline void mcmf()
{
int flow=0;
int cost=0;
while(spfa())
{
flow+=change[t];
cost+=change[t]*dis[t];
for(int i=t;i!=s;i=e[road[i]^1].to)
{
e[road[i]].can-=change[t];
e[road[i]^1].can+=change[t];
}
}
if(cost==2)
{
printf("2\n");
std::cout<<f[1]<<'\n'<<f[n]<<'\n'<<f[1];
}
else if(flow==2)
{
printf("%d\n",cost);
for(int i=1;i<=v;i++)
for(int j=head[aa[i]+n];j;j=e[j].nxt)
if(e[j].to==bb[i]&&!e[j].can)
add(aa[i],bb[i]),add(bb[i],aa[i]);
dfs(s);
for(int i=1;i<=top;i++) std::cout<<f[st[i]]<<'\n';
std::cout<<f[1];
}
else return (void)(printf("No Solution!"));
}
int main()
{
n=in(),v=in();
s=1,t=n*2;
for(int i=1;i<=n;i++)
{
std::cin>>f[i];
mp[f[i]]=i;
link(i,i+n,1,1);
}
link(s,s+n,0,1);
link(n,n+n,0,1);
for(int i=1;i<=v;i++)
{
std::cin>>ss1>>ss2;
if(mp[ss1]>mp[ss2]) std::swap(ss1,ss2);
aa[i]=mp[ss1],bb[i]=mp[ss2];
link(mp[ss1]+n,mp[ss2],0,1);
}
mcmf();
return 0;
}

P2770 航空路线问题的更多相关文章

  1. 网络流 P2770 航空路线问题

    #include <cstdio> #include <cstdlib> #include <map> #include <queue> #includ ...

  2. Luogu P2770 航空路线问题

    题目链接 \(Click\) \(Here\) 本来想调剂心情没想到写了那么久,还被\(dreagonm\)神仙嘲讽不会传纸条,我真是太弱了\(QAQ\)(原因:最开始写最大费用最大流一直想消圈,最后 ...

  3. 洛谷P2770 航空路线问题(费用流)

    传送门 完了这题好厉害……字符串什么的好麻烦…… 要求从$1$到$n$的路径,不重复,经过边数最多 每一个点拆成两个,$A_i,B_i$,然后$A_i$到$B_i$连容量为$1$,费用为$1$的边,保 ...

  4. 洛谷P2770 航空路线问题(费用流)

    题意 $n$个点从左向右依次排列,有$m$条双向道路 问从起点到终点,再从终点回到起点,在经过的点不同的情况下最多能经过几个点 Sol 首先,问题可以转化为求两条互不相交的路径,使得点数最多 为了满足 ...

  5. 洛谷 P2770 航空路线问题【最大费用最大流】

    记得cnt=1!!因为是无向图所以可以把回来的路看成另一条向东的路.字符串用map处理即可.拆点限制流量,除了1和n是(i,i+n,2)表示可以经过两次,其他点都拆成(i,i+n,1),费用设为1,原 ...

  6. 洛谷P2770 航空路线问题 最小费用流

    Code: #include<cstdio> #include<iostream> #include<algorithm> #include<vector&g ...

  7. 【题解】【网络流24题】航空路线问题 [P2770] [Loj6122]

    [题解][网络流24题]航空路线问题 [P2770] [Loj6122] 传送门:航空路线问题 \([P2770]\) \([Loj6122]\) [题目描述] 给出一张有向图,每个点(除了起点 \( ...

  8. JS前端三维地球渲染——中国各城市航空路线展示

    前言 我还从来没有写过有关纯JS的文章(上次的矢量瓦片展示除外,相对较简单.),自己也学习过JS.CSS等前端知识,了解JQuery.React等框架,但是自己艺术天分实在不过关,不太喜欢前端设计,比 ...

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

    #6122. 「网络流 24 题」航空路线问题 题目描述 给定一张航空图,图中顶点代表城市,边代表两个城市间的直通航线.现要求找出一条满足下述限制条件的且途经城市最多的旅行路线. 从最西端城市出发,单 ...

随机推荐

  1. pos机的热敏纸尺寸

    57x50或者是57x30,两个型号宽度都是一样的,只是厚度不一样,前者是厚一点,适合固定机用,后者适合移动POS机用 厚度不是指纸的厚度,而是纸的容量,移动机的纸槽较小只能用57X30的

  2. 此上下文中不允许异步操作。启动异步操作的页必须将 Async 特性设置为 true,并且异步操作只能在 PreRenderComplete 事件之前的页上启动。

    <%@ Page Language="C#" AutoEventWireup="true" ...... Async="true" % ...

  3. 一个hitbernate配置文件,带几个表(注意mapping);如果连接字符串没有设置utf-8,向insert mysql 会产生乱码(utf8 或 utf-8)

    <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-configuratio ...

  4. Tiny4412 u-boot分析(2)u-boot启动流程

    从大方面来说,u-boot的启动分成两个阶段,第一个阶段主要的职责是准备初始化的环境,主要有以下几点 ①设置异常向量表 ②把CPU的工作模式设置为SVC32模式 ③关闭中断.MMU和cache ④关闭 ...

  5. js兼容事件

    //浏览器检测(function () { window.sys = {}; var ua = navigator.userAgent.toLowerCase(); var s; (s = ua.ma ...

  6. 对数组名取地址 a[ ],&a

    C语言规定,数组名代表数组的首地址,也就是第0号元素的地址.所以a==&a[0] 但对数组名取地址时却要注意了,在理解“对数组名取地址”这一表达式的含义时一定要记住:数组名是“数组”这种变量的 ...

  7. Hibernate和JPA

    ORM(Object/Relational Mapping : 对象关系映射)就是利用描述对象和数据库之间映射的元数据,自动(且透明)的将java应用程序中的对象持久化到关系数据库的表中.HIbern ...

  8. nodejs读文件

    1.异步读取文件:var fs= require('fs'); // 从文件系统中读取请求的文件内容 fs.readFile(pathname.substr(1), function (err, da ...

  9. cximage使用

    为什么使用CxImage u 使用简单:下面的入门教程会给你展示这一点. 构造函数 初始化 多帧图像操作 区域选择操作 Alpha通道操作 Layers 图像层操作 Attributes 图像属性操作 ...

  10. Java面试问题列表