PAT (Advanced Level) Practice - 1131 Subway Map(30 分)

简介: PAT (Advanced Level) Practice - 1131 Subway Map(30 分)

题目链接:点击打开链接

题目大意:找出一条路线,使得对任何给定的起点和终点,可以找出中途经停站最少的路线;如果经停站一样多,则取需要换乘线路次数最少的路线。

解题思路:一遍DFS即可~DFS过程中要维护两个变量:minCnt-中途经停的最少的站; minTransfer-需要换乘的最小次数~

  1. 可以这样计算出一条线路的换乘次数:在line[10000][10000]的数组中保存每两个相邻站中间的线路是几号线~从头到尾遍历最终保存的路径,preLine为前一小段的线路编号,如果当前的结点和前一个结点组成的这条路的线路编号和preLine不同,说明有一个换乘,就将cnt+1,最后遍历完累加的cnt即是换乘的次数~
  2. update:由于新的PAT系统中会显示原解法有一个测试用例内存超限,考虑到line[10000][10000]存储的方式太暴力了,所以将line换成了unordered_map<int, int> line存储的方式,第一个int用来存储线路,每次将前四位存储第一个线路,后四位存储第二个线路,使用a[i-1]*10000+a[i]的方式存储,第二个int用来保存两个相邻中间的线路是几号线~
  3. 可以这样计算出一条线路中途停站的次数:在dfs的时候有个变量cnt,表示当前路线是所需乘的第几个站,每次dfs时候将cnt+1表示向下遍历一层~cnt就是当前中途停站的次数~
  4. 可以这样输出结果:和计算线路换乘次数的思路一样,每当preLine和当前Line值不同的时候就输出一句话~保存preTransfer表示上一个换乘站,最后不要忘记输出preTransfer和最后一个站之间的路即使最后一个站并不是换乘站~。
  5. 有时候,全局变量对 DFS 是禁忌,特别是每次回溯时需要变回起初的值的变量,如果用全局替代局部变量的话,一旦改变就回不去起初的“还原点”了。


AC 代码

#include<bits/stdc++.h>#include<cmath>#define mem(a,b) memset(a,b,sizeof a)#define ssclr(ss) ss.clear(), ss.str("")#define INF 0x3f3f3f3f#define MOD 1000000007usingnamespacestd;
typedeflonglongll;
constintmaxn=1e4+10;
intd, id;
intminCnt, minTran;
intvis[maxn];
unordered_map<int,int>line;
vector<int>tpath, path, v[maxn];
intfhash(intid1,intid2)
{
returnid1*10000+id2;
}
intcalTran()
{
intpreline=0, cnt=-1;
for(inti=1;i<tpath.size();i++)
    {
id=fhash(tpath[i-1],tpath[i]);
if(line[id]!=preline) cnt++, preline=line[id];
    }
returncnt;
}
voiddfs(ints, intcnt)
{
if(s==d&& (cnt<minCnt||cnt==minCnt&&calTran()<minTran))
    {
path=tpath;
minCnt=cnt;
minTran=calTran();
    }
if(s==d) return;
for(inti=0;i<v[s].size();i++)
    {
inttid=v[s][i]; // 去掉 int dfs最终答案会出问题,因为当 tid 作为全局变量时,一直在变,而局部变量对dfs的一个很大的作用就是每次回溯时,该变量的值可以恢复到起初的值if(!vis[tid])
        {
vis[tid]=1;
tpath.push_back(tid);
dfs(tid, cnt+1);
vis[tid]=0;
tpath.pop_back();
        }
    }
}
intmain()
{
intn,m,pre,q,s;
scanf("%d",&n);
for(inti=1;i<=n;i++)
    {
scanf("%d%d",&m,&pre);
for(intj=1;j<m;j++)
        {
scanf("%d",&id);
v[pre].push_back(id);
v[id].push_back(pre);
line[fhash(pre,id)]=line[fhash(id,pre)]=i;
pre=id;
        }
    }
scanf("%d",&q);
while(q--)
    {
scanf("%d%d",&s,&d);
mem(vis,0), tpath.clear(), minCnt=minTran=INF;
vis[s]=1;
tpath.push_back(s);
dfs(s,0);
tpath.pop_back();
vis[s]=0;
printf("%d\n",minCnt);
intpreline=0, len=path.size(), preTran;
for(inti=1;i<len;i++)
        {
id=fhash(path[i-1],path[i]);
if(line[id]!=preline)
            {
if(preline!=0) printf("Take Line#%d from %04d to %04d.\n",preline,preTran,path[i-1]);
preline=line[id];
preTran=path[i-1];
            }
        }
printf("Take Line#%d from %04d to %04d.\n",preline,preTran,path[len-1]);
    }
return0;
}
目录
相关文章
PAT (Advanced Level) Practice - 1111 Online Map(30 分)
PAT (Advanced Level) Practice - 1111 Online Map(30 分)
122 0
|
7月前
|
Dart
Dart之集合详解(List、Set、Map)
Dart之集合详解(List、Set、Map)
|
7月前
|
存储 JavaScript 前端开发
JavaScript进阶-Map与Set集合
【6月更文挑战第20天】JavaScript的ES6引入了`Map`和`Set`,它们是高效处理集合数据的工具。`Map`允许任何类型的键,提供唯一键值对;`Set`存储唯一值。使用`Map`时,注意键可以非字符串,用`has`检查键存在。`Set`常用于数组去重,如`[...new Set(array)]`。了解它们的高级应用,如结构转换和高效查询,能提升代码质量。别忘了`WeakMap`用于弱引用键,防止内存泄漏。实践使用以加深理解。
86 3
|
4月前
|
Go 定位技术 索引
Go 语言Map(集合) | 19
Go 语言Map(集合) | 19
|
4月前
|
存储 前端开发 API
ES6的Set和Map你都知道吗?一文了解集合和字典在前端中的应用
该文章详细介绍了ES6中Set和Map数据结构的特性和使用方法,并探讨了它们在前端开发中的具体应用,包括如何利用这些数据结构来解决常见的编程问题。
ES6的Set和Map你都知道吗?一文了解集合和字典在前端中的应用
|
5月前
|
存储 安全 Java
java集合框架复习----(4)Map、List、set
这篇文章是Java集合框架的复习总结,重点介绍了Map集合的特点和HashMap的使用,以及Collections工具类的使用示例,同时回顾了List、Set和Map集合的概念和特点,以及Collection工具类的作用。
java集合框架复习----(4)Map、List、set
|
5月前
|
Java
【Java集合类面试二十二】、Map和Set有什么区别?
该CSDN博客文章讨论了Map和Set的区别,但提供的内容摘要并未直接解释这两种集合类型的差异。通常,Map是一种键值对集合,提供通过键快速检索值的能力,而Set是一个不允许重复元素的集合。
|
5月前
|
算法 Java 索引
【Java集合类面试四】、 描述一下Map put的过程
这篇文章详细描述了HashMap中put操作的过程,包括首次扩容、计算索引、插入数据以及链表转红黑树和可能的再次扩容。
【Java集合类面试四】、 描述一下Map put的过程
|
5月前
|
存储
|
5月前
|
安全 Java
【Java集合类面试五】、 如何得到一个线程安全的Map?
如何得到一个线程安全的Map的方法包括:使用Collections工具类将Map包装为线程安全,使用java.util.concurrent包下的ConcurrentHashMap,以及不推荐使用性能较差的Hashtable。