题目分析:

本题我有两种思路,一种是只依靠dijkstra算法,在dijkstra部分直接判断所有的情况,以局部最优解得到全局最优解,另一种是dijkstra + dfs,先计算出最短距离以及每个点的可能前驱点,然后用dfs搜索每一条道路对最优路径进行维护,并且第二种方法记录道路的方式比较巧妙值得学习掌握(在dfs部分用一条临时路径进行维护)

对于字符串如何以整数的形式存储到二维数组中,这里用的是map的方式,当然也可以通过字符串计算出hash值去索引,毕竟是三个大写字母的字符串

本题代码:

  1 #include<iostream>
2 #include<string>
3 #include<cmath>
4 #include<map>
5 #include<string.h>
6 #include<vector>
7 #include<algorithm>
8 #include<stdio.h>
9 using namespace std;
10
11 const int M = 0x3f3f3f3f;
12 const int N = 205;
13 int mat[N][N];
14 int vis[N];
15 int dist[N];
16 int peo[N];
17 vector<int> pre[N];
18 vector<int> path, temppath;
19 map<string, int> mp1; //每个城市对应的id
20 map<int, string> mp2; //每个id对应的城市
21 int n, m, min_dist, max_kill, road_num;
22 string from, to;
23
24 int minn(){
25 int k = -1;
26 int Min = M;
27 for(int i = 1; i <= n; i++){
28 if(vis[i] == 0 && dist[i] < Min){
29 Min = dist[i];
30 k = i;
31 }
32 }
33 return k;
34 }
35
36 void dijkstra(){
37 max_kill = 0;
38 road_num = 0;
39 min_dist = 0;
40 memset(vis, 0, sizeof(vis));
41 memset(dist, M, sizeof(dist));
42 dist[1] = 0; //起点到自己距离为0
43 for(int i = 1; i <= n; i++){
44 int k = minn();
45 if(k == -1) break;
46 vis[k] = 1;
47 for(int j = 1; j <= n; j++){
48 if(vis[j] == 0 && dist[k] + mat[k][j] < dist[j]){
49 dist[j] = dist[k] + mat[k][j];
50 pre[j].clear();
51 pre[j].push_back(k);
52 }else if(vis[j] == 0 && dist[k] + mat[k][j] == dist[j]){
53 pre[j].push_back(k);
54 }
55 }
56 }
57 min_dist = dist[mp1[to]];
58 }
59
60 void dfs(int x){
61 temppath.push_back(x);
62 if(x == mp1[from]){
63 //现在需要找出同距离下点最多且杀敌最多
64 road_num++; //不论是点多的 少的 一样多的都算是同样距离的一条最短路径
65 int kill = 0;
66 for(int i = temppath.size()-1; i >= 0; i--){
67 int id = temppath[i];
68 kill += peo[id];
69 }
70 if(temppath.size() > path.size()){
71 path = temppath;
72 max_kill = kill;
73 }else if(temppath.size() == path.size()){
74 if(kill > max_kill){
75 max_kill = kill;
76 path = temppath;
77 }
78 }
79 temppath.pop_back();
80 return;
81 }
82 for(int i = 0; i < pre[x].size(); i++)
83 dfs(pre[x][i]);
84 temppath.pop_back();
85 }
86
87 int main(){
88 scanf("%d%d", &n, &m);
89 cin>>from>>to;
90 mp1[from] = 1;
91 mp2[1] = from;
92 for(int i = 2; i <= n; i++){
93 string s; int x;
94 cin>>s; scanf("%d", &x);
95 mp1[s] = i;
96 mp2[i] = s;
97 peo[mp1[s]] = x;
98 }
99 memset(mat, M, sizeof(mat));
100 for(int i = 1; i <= m; i++){
101 string s1, s2;
102 cin>>s1>>s2;
103 scanf("%d", &mat[mp1[s1]][mp1[s2]]);
104 mat[mp1[s2]][mp1[s1]] = mat[mp1[s1]][mp1[s2]];
105 }
106 dijkstra(); //计算出最短路径
107 dfs(mp1[to]); //计算最优路径
108 cout<<mp2[1];
109 for(int i = path.size()-2; i >= 0; i--) cout<<"->"<<mp2[path[i]]; //注意起始点已经输出了
110 printf("\n");
111 printf("%d %d %d\n", road_num, min_dist, max_kill);
112 return 0;
113 }

天梯赛练习 L3-011 直捣黄龙 (30分) dijkstra + dfs的更多相关文章

  1. 天梯赛练习 L3-008 喊山 (30分) bfs搜索

    题目分析: 本题是一题比较简单的bfs搜索题,首先由于数据给的比较多不能直接开二维数组存放,而是用了vector的动态的二维数组的形式存放,对于每个出发点,我们bfs向四周搜索,标记搜索过的点,遇到搜 ...

  2. PAT天梯赛练习 L3-004 肿瘤诊断 (30分) 三维BFS

    题目分析: 可能是我的理解能力比较差,在读题的时候一直以为所有的切片是可以排列组合的,并不是按照输入顺序就定死的,那么这题就变得十分的复杂啦~~~~~,查看的题解之后发现所有的切片并没有所谓的自由组合 ...

  3. L3-007 天梯地图 (30 分) dijkstra

    本题要求你实现一个天梯赛专属在线地图,队员输入自己学校所在地和赛场地点后,该地图应该推荐两条路线:一条是最快到达路线:一条是最短距离的路线.题目保证对任意的查询请求,地图上都至少存在一条可达路线. 输 ...

  4. PAT 甲级 1053 Path of Equal Weight (30 分)(dfs,vector内元素排序,有一小坑点)

    1053 Path of Equal Weight (30 分)   Given a non-empty tree with root R, and with weight W​i​​ assigne ...

  5. 天梯赛练习 L3-007 天梯地图 (30分) Dijkstra

    题目分析: 本题的题意比较清晰,就是有一个起点和一个终点,给出m条路径,可能是单向的可能是双向的,同时一条路有两个权重,分别是通过这条路需要的时间和这条路的路径长度,题目需要求出两条路径,一条是在最快 ...

  6. 团体程序设计天梯赛L3-019 代码排版(23分)

    打算学完编译原理后再次实现它... 以下为比较“杂乱”的方法: 海量数据: https://pan.baidu.com/s/1Prd0ZqNLoCLLvXyJjCef3w 如果大家有发现这个程序的问题 ...

  7. 【PAT甲级】1053 Path of Equal Weight (30 分)(DFS)

    题意: 输入三个正整数N,M,S(N<=100,M<N,S<=2^30)分别代表数的结点个数,非叶子结点个数和需要查询的值,接下来输入N个正整数(<1000)代表每个结点的权重 ...

  8. 1034 Head of a Gang (30分)(dfs 利用map)

    One way that the police finds the head of a gang is to check people's phone calls. If there is a pho ...

  9. PAT A1103 Integer Factorization (30 分)——dfs,递归

    The K−P factorization of a positive integer N is to write N as the sum of the P-th power of K positi ...

随机推荐

  1. Vue:对axios进行简单的二次封装

    主要做3点: 1.配置一个请求地址前缀 2.请求拦截(在请求发出去之前拦截),给所有的请求都带上 token 3.拦截响应,遇到 token 不合法则报错 // 对 axios 的二次封装 impor ...

  2. EditPlus各个版本的注册码,亲测可用

    原文链接:https://www.cnblogs.com/shihaiming/p/6422441.html EditPlus4注册码 注册名:host1991    序列号:14F50-CD5C8- ...

  3. Day4 dict和set

    dict  -- dictionary    一组key的集合,包含key与value的对应.        Python内置的字典,在其他语言中称为map,使用key-value存储,具有极快的查找 ...

  4. js下 Day01、DOM对象,BOM浏览器对象模型

    一.初识DOM 1.什么是DOM?为什么学习DOM 2.DOM是实现js在网页实现交互的关键环节,我们的js代码就是通过DOM的方法来实现对于html内容的操作. 3.认识DOM实现了js和网页结合的 ...

  5. Spark性能调优九之常用算子调优

    1.使用mapPartitions算子提高性能 mapPartition的优点:使用普通的map操作,假设一个partition中有1万条数据,那么function就要被执行1万次,但是使用mapPa ...

  6. Web服务器-正则表达式-整理+复习(3.1.1)

    @ 目录 1.常用api 2.例子 3.正则表达式模式 关于作者 参考文章 1.常用api re.match(pattern, string, flags=0) pattern 匹配的正则表达式 st ...

  7. 多任务-python实现-进程pool(2.1.9)

    @ 目录 1. 概念 2.python代码实现 1. 概念 当需要创建的子进程数量不多时,可以直接利用multiprocessing中的Process动态成生多个进程,但如果是上百甚至上千个目标,手动 ...

  8. 开源项目葫芦藤:IdentityServer4的实现及其运用

    目录 前言 签名证书(Signing Credential) 客户端存储(Client Store) 资源存储(Resource Store) 持久化授权存储(Persisted Grant Stor ...

  9. solidworks 2018 因动态绘制边线显示视图延迟的解决方案

    每次鼠标移动到一个物体上时总是会卡顿几秒,直到完成所有边线的绘制后才可以继续进行其他操作,这体验实在是不好. 解决方案很简单,只要取消这个默认开启的动态高亮显示就可以了. 1.去 选项->系统选 ...

  10. python之列表操作的几个函数

    Python中的列表是可变的,这是它却别于元组和字符串最重要的特点,元组和字符串的元素不可修改.列举一些常用的列表操作的函数和方法. 1,list.append(x),将x追加到列表list末尾: 1 ...