Toll! Revisited

大致题意:有两种节点,一种是大写字母,一种是小写字母。

首先输入m条边。当经过小写字母时须要付一单位的过路费。当经过大写字母时,要付当前財务的1/20做过路费。

问在起点最少须要带多少物品使到达终点时还有k个物品。

当有多条符合条件的路径时输出字典序最小的一个。



思路:已知终点的权值,那么能够从终点向前推。

求终点到起点的最短路径,然后按字典序打印路径。

比較难处理的是:向前推时前驱节点的权值计算。列个方程算算就能够了,主要时不能整除的情况。

计算前驱结点dis值的时候,同一时候记录(i,j)的边权值。这是打印路径的根据。



#include <stdio.h>
#include <algorithm>
#include <set>
#include <map>
#include <vector>
#include <math.h>
#include <string.h>
#include <stack>
#include <queue>
#define LL long long
#define _LL __int64 using namespace std;
const int INF = 0x3f3f3f3f;
const int maxm = 1010;
const int maxn = 60; struct node
{
int v,w;
int next;
}edge[maxm]; int m;
int pre[maxn],cnt;
int start,end;
LL dis[maxn],p;
int vis[maxn];
vector <int> ans; void init()
{
cnt = 0;
memset(pre,-1,sizeof(pre));
for(int i = 0; i < maxm; i++)
edge[i].w = 0;
} void add(int u, int v)
{
edge[cnt].v = v;
edge[cnt].next = pre[u];
pre[u] = cnt++;
} void dijstra()
{
priority_queue <pair<LL,int>, vector<pair<LL,int> >, greater<pair<LL,int> > > que;
while(!que.empty()) que.pop();
memset(dis,INF,sizeof(dis));
memset(vis,0,sizeof(vis)); dis[end] = p; que.push(make_pair(dis[end],end)); while(!que.empty())
{
int u = que.top().second;
que.pop();
if(vis[u]) continue;
vis[u] = 1; for(int i = pre[u]; i != -1; i = edge[i].next) //松弛相邻节点
{
if(vis[edge[i].v]) continue;
int v = edge[i].v; if(u < 26)
{
//计算前驱结点的权值,推断是否整除,若不整除,尝试加1继续推断
if(dis[u]%19 == 0)
{
if(dis[v] > dis[u]/19*20)
{
dis[v] = dis[u]/19*20;
edge[i].w = edge[i^1].w = dis[v]-dis[u];
que.push(make_pair(dis[v],v));
}
}
else if( (dis[u]+1)%19 )
{
if(dis[v] > (dis[u]+1)*20/19)
{
dis[v] = (dis[u]+1)*20/19;
edge[i].w = edge[i^1].w = dis[v]-dis[u];
que.push(make_pair(dis[v],v));
}
}
else
{
if(dis[v] > (dis[u]+1)*20/19-1 )
{
dis[v] = (dis[u]+1)*20/19-1;
edge[i].w = edge[i^1].w = dis[v]-dis[u];
que.push(make_pair(dis[v],v));
}
}
}
else
{
if(dis[v] > dis[u]+1)
{
dis[v] = dis[u]+1;
edge[i].w = edge[i^1].w = 1;
que.push(make_pair(dis[v],v));
}
} }
}
} void solve()
{
ans.clear();
int now;
now = start;
ans.push_back(now);
while(now != end)
{
int tmp = 1<<6;
for(int i = pre[now]; i != -1; i = edge[i].next)
{
int v = edge[i].v;
//由于输出字典序最小的,所以求出满足dis[now] - dis[v] == edge[i].w中最小的v
if(dis[now] - dis[v] == edge[i].w && v < tmp)
{
tmp = v;
}
}
ans.push_back(tmp);
now = tmp;
} printf("%c",ans[0]+'A'); for(int i = 1; i < (int)ans.size(); i++)
printf("-%c",ans[i]+'A');
printf("\n");
}
int main()
{
int item = 1;
char t1,t2;
while(~scanf("%d",&m))
{
if(m == -1) break;
init();
getchar();
for(int i = 0; i < m; i++)
{
scanf("%c %c",&t1,&t2);
getchar();
add(t1-'A',t2-'A');
add(t2-'A',t1-'A');
} scanf("%lld %c %c",&p,&t1,&t2);
start = t1-'A';
end = t2-'A'; dijstra();
printf("Case %d:\n",item++);
printf("%lld\n",dis[start]);
solve();
}
return 0;
}



uva 10537 Toll! Revisited(优先队列优化dijstra及变形)的更多相关文章

  1. UVA 10537 Toll! Revisited (逆推,最短路)

    从终点逆推,d[u]表示进入u以后剩下的货物,那么进入u之前的货物数量设为y,d[u] = x,那么y-x=ceil(y/20.0)=(y-1)/20+1=(y+19)/20. (y-x)*20+r= ...

  2. UVA 10537 - The Toll! Revisited(dijstra扩张)

    UVA 10537 - The Toll! Revisited option=com_onlinejudge&Itemid=8&page=show_problem&catego ...

  3. 地铁 Dijkstra(优先队列优化) 湖南省第12届省赛

    传送门:地铁 思路:拆点,最短路:拆点比较复杂,所以对边进行最短路,spfa会tle,所以改用Dijkstra(优先队列优化) 模板 /******************************** ...

  4. 最短路算法模板合集(Dijkstar,Dijkstar(优先队列优化), 多源最短路Floyd)

    再开始前我们先普及一下简单的图论知识 图的保存: 1.邻接矩阵. G[maxn][maxn]; 2.邻接表 邻接表我们有两种方式 (1)vector< Node > G[maxn]; 这个 ...

  5. UVA10537 Toll! Revisited

    difkstra + 路径输出 The Toll! Revisited Time Limit: 3000MS Memory Limit: Unknown 64bit IO Format: %lld & ...

  6. 堆优化dijstra

    因为spfa没事就被卡一卡,所以堆优化dijstra就显得很重要,在最短路或者其模型里边,最少有一条边是没有被更新过的,也就是它是最短的,同理从这个点开始也有一条边最短,所以每次就找最短的然后松弛操作 ...

  7. dijkstra 的优先队列优化

    既然要学习算法,就要学习到它的精髓,才能够使用起来得心应手. 我还是远远不够啊. 早就知道,dijkstra 算法可以用优先队列优化,我却一直不知道该怎样优化.当时,我的思路是这样的:假设有n个顶点, ...

  8. poj 1511 优先队列优化dijkstra *

    题意:两遍最短路 链接:点我 注意结果用long long #include<cstdio> #include<iostream> #include<algorithm& ...

  9. 最短路--dijkstra+优先队列优化模板

    不写普通模板了,还是需要优先队列优化的昂 #include<stdio.h> //基本需要的头文件 #include<string.h> #include<queue&g ...

随机推荐

  1. Python访问MySQL数据库

    #encoding: utf-8 import mysql.connector __author__ = 'Administrator' config={'host':'127.0.0.1',#默认1 ...

  2. rapidxml 序列化

    void TestRapidXml() { ]; sprintf(xmlContent,"<root><head>aaa</head><body&g ...

  3. 从零开始学JavaScript四(数据类型)

    一.分类 基本数据类型:undefined.null.string.Boolean.number 复杂数据类型:object object的属性以无序的名称和值对的形式 (name : value) ...

  4. HTTP Content-type整理

    文件扩展名 Content-Type(Mime-Type) 文件扩展名 Content-Type(Mime-Type) .*( 二进制流.不知道下载文件类型) application/octet-st ...

  5. java统计abacbacdadbc中的每个字母出现的次数,输出格式是:a(4)b(3)c(3)d(2)

    import java.util.Iterator; import java.util.Map; import java.util.Set; import java.util.TreeMap; /* ...

  6. 28种CSS3炫酷载入动画特效

    这是一组效果很炫酷的纯CSS3 Loading载入动画特效.这组loading动画共同拥有27种不同的效果.每一种loading动画都是通过CSS3的keyframes帧动画来完毕的,每个载入动画都构 ...

  7. (39)JS运动之缓冲运动

    基本思路:使用定时器让物体向右运动,在运动的过程中再不是匀速运动,而是先快后慢,即距离越大,速度越快,距离越小,速度越小,可是到达终点的时候,必须注意要使用向上取整函数Math.ceil()和向下取整 ...

  8. mysql高效索引之覆盖索引

    概念 如果索引包含所有满足查询需要的数据的索引成为覆盖索引(Covering Index),也就是平时所说的不需要回表操作 判断标准 使用explain,可以通过输出的extra列来判断,对于一个索引 ...

  9. supervisor介绍与安装

    前言 今天同事让我帮忙安装一个叫supervisor的软件,但自己确实没接触过这个软件 自己做一下学习的记录 我首先是查询了一下supervisor的官网,初步认识一下这个软件 Supervisor是 ...

  10. 两个有序数组求中位数log(m+n)复杂度

    leetcode 第4题 中位数技巧: 对于长度为L的有序数组,它的中位数是(a[ceil((L+1)/2)]+a[floor((L+1)/2)])/2 算法原理: 类似三分法求极值 两个人都前进,谁 ...