【1087】All Roads Lead to Rome (30 分)

简介: 【1087】All Roads Lead to Rome (30 分)【1087】All Roads Lead to Rome (30 分)
#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<math.h>
#include<string.h>
#include<string>
#include<algorithm>  
#include<map>
#include<vector>
#include<queue> 
using namespace std;  
dij找出所有最短路径,DFS筛选出同条件下点权之和、平均点权min的路径
注意用map建立城市字符和编号映射
const int MAXV=210;//最大顶点数
const int INF=1000000000;  //无穷大
//n为顶点数,m为边数,st为起点,G为邻接矩阵,weight为点权(幸福值)
//d[]记录最短距离,numPath记录最短路径条数
//maxW记录最大点权之和,maxAvg为最大平均点权
int n,m,st,G[MAXV][MAXV],weight[MAXV];
int d[MAXV],numPath=0,maxW=0;
double maxAvg=0;
bool vis[MAXV]={false}; //vis[i]=true表示顶点i已访问,初始均为false
vector<int> pre[MAXV];//前驱
vector<int> tempPath,path;  //临时路径及最优路径
map<string,int> cityToIndex;  //将城市名转换为编号
map<int,string> indexToCity;  //将编号转换为城市名
void Dijkstra(int s){  //s为起点
  fill(d,d+MAXV,INF);  //初始化起点到每个结点的距离都为无穷大
  d[s]=0; //起点到自己的距离为0
  for(int i=0;i<n;i++){  //循环n次
    int u=-1,MIN=INF; //u使d[u],MIN存放该最小的d[u]
    for(int j=0;j<n;j++){ //找到未访问的顶点中d[]中最小的
      if(vis[j]==false && d[j]<MIN){
        u=j;
        MIN=d[j];
      }
    }
    //找不到小于INF的d[u],说明剩下的顶点和起点s不连通
    if(u==-1)  return;
    vis[u]=true;//标记u为已访问
    for(int v=0;v<n;v++){
      //如果v为访问 && u能够到达v
      if(vis[v] == false && G[u][v] !=INF){
        if(d[u]+G[u][v] < d[v]){ //以u为中介点使d[v]更小
          d[v]=d[u]+G[u][v];  //优化d[v]
          pre[v].clear() ; //清空pre[v]
          pre[v].push_back(u);  //u为v的前驱
        }else if(d[u]+G[u][v] == d[v]){  //找到相同长度的路径
          pre[v].push_back(u); //u为v的前驱之一
        }
      }
    }
  }
}
void DFS(int v){ //v为当前结点
  if(v==st){ //递归边界,到达叶子结点(路径起点)
    tempPath.push_back(v);
    numPath++;  //最短路径条数加1
    int tempW=0; //临时路径tempPath的点权之和
    for(int i=tempPath.size()-2 ; i>=0;i--){ //倒着访问
      int id=tempPath[i]; //当前结点
      tempW +=weight[id];  //增加点id的点券(注意不是i)
    }
    double tempAvg=1.0*tempW / (tempPath.size()-1);  //临时平均点权
    if(tempW >maxW){  //如果当前路径的点权之和更大
      maxW=tempW;  
      maxAvg=tempAvg;  
      path=tempPath;
    }else if(tempW==maxW && tempAvg> maxAvg){
      //点权之和相同,平均点权更大
      maxAvg=tempAvg; 
      path=tempPath;
    }
    tempPath.pop_back();
    return;
  }
  //以下为递归式
  tempPath.push_back(v);//将当前访问结点加入临时路径tempPath的最后面
  for(int i=0;i<pre[v].size();i++){
    DFS(pre[v][i]); //结点v的前驱结点pre[v][i],递归
  }
  tempPath.pop_back(); //遍历完当前所有前驱结点,将当前结点v删除
}
int main(){   
  string start,city1,city2;
  cin >> n >> m >> start; //城市数  路径数  起点
  cityToIndex[start]=0;//将起始城市字符映射成 0
  indexToCity[0]=start; //将0映射成相应的起始城市字符
  for(int i=1;i<=n-1;i++){
    cin >> city1 >> weight[i]; //保存每个城市的幸福值
    cityToIndex[city1]=i;
    indexToCity[i]=city1;
  }
  fill(G[0],G[0]+MAXV*MAXV,INF);  //初始化图G
  for(int i=0;i<m;i++){ 
    cin>>city1>>city2;
    int c1=cityToIndex[city1],c2=cityToIndex[city2]; 
    cin >> G[c1][c2];
    G[c2][c1]=G[c1][c2];
  }
  Dijkstra(0); 
  int rom=cityToIndex["ROM"]; //终点城市字符转换成对应数字
  DFS(rom); //获取最优路径
  printf("%d %d %d %d\n",numPath,d[rom],maxW,(int)maxAvg); //maxW为最大点权之和
  for(int i=path.size()-1;i>=0;i--){  
    cout << indexToCity[path[i]]; //倒着输出路径上的结点
    if(i>0)  cout << "->";
  }
  system("pause"); 
    return 0;   
}
相关文章
codeforces 347A - Difference Row
给你一个序列,让你求(x1 - x2) + (x2 - x3) + ... + (xn - 1 - xn).值最大的一个序列,我们化简一下公式就会发现(x1 - x2) + (x2 - x3) + ... + (xn - 1 - xn). = x1 - xn, 也就是说只有第一个和最后一个是确定的,其他的随便了! 也不是了, 还要让你按字典序最小的排列,也就是说其他的是按飞递减序排列的,简单的一次排序就OK了。
34 0
|
存储 机器学习/深度学习 算法
【PAT甲级】1087 All Roads Lead to Rome
【PAT甲级】1087 All Roads Lead to Rome
82 0
|
SQL Oracle 关系型数据库
Oracle的LAG和LEAD分析函数
Oracle的LAG和LEAD分析函数
383 1
Oracle的LAG和LEAD分析函数
LeetCode 115. Distinct Subsequences
给定字符串S和字符串T,计算S的不同子序列的数量,使其其等于T. 字符串的子序列是一个新字符串,它是通过删除一些(可以是无)字符而不干扰其余字符的相对位置而从原始字符串形成的。 (即,“ACE”是“ABCDE”的子序列,而“AEC”不是)。
74 0
LeetCode 115. Distinct Subsequences
|
移动开发 前端开发
ICPC Latin American Regional 2017-Imperial roads(LCA)
题目大意: 给出n个点,m条边的一个图,q个询问, 每次询问给出两个点u,v,问包含u-v这条边的最小生成树是多少 这道题比较板 首先求一下这个图的最小生成树对于这n个点,最小生成树一定是n-1条边,如果说再加上一条边,一定会构成一个环。 我们把生成的这个最小生成树看作是一个以1为根节点的最小生成树。 所以说在下面的q个询问中,如果说这条边用到了最小生成树中(这条边是最小生成树上的边),那么直接输出当前最小生成树的代价就好;如果说当前这条边没有出现在最小生成树当中,那么最小生成树的权值val加上这条边之后就构成了一个环,求出这两个点所在的环内的最大边权,并将这个边权减去,就是最终结果
123 0
ICPC Latin American Regional 2017-Imperial roads(LCA)
PAT (Advanced Level) Practice - 1087 All Roads Lead to Rome(30 分)
PAT (Advanced Level) Practice - 1087 All Roads Lead to Rome(30 分)
110 0
【1030】Travel Plan (30 分)
【1030】Travel Plan (30 分) 【1030】Travel Plan (30 分)
111 0
【1096】Consecutive Factors (20 分)
【1096】Consecutive Factors (20 分) 【1096】Consecutive Factors (20 分)
133 0
|
算法 JavaScript Java
并查集算法 - Algorithms, Part I, week 1 UNION-FIND
cousera 普林斯顿大学 算法公开课 第一周 并查集数据类型内容整理
1524 0

热门文章

最新文章