Full Tank

题目大意

给定一张 \(n\) 个点,\(m\) 条边的连通无向图,在每个点有一个加油站,油价为该点的点权,每条边的油耗为该边的边权。现给出若干询问,问一辆油箱容量为 \(c\) 的车子是否能从 \(s\) 开到 \(e\),如果可以,求出最小所需的钱。

思路分析

看到这种有图有权求最小消耗的题,我们首先考虑最短路。

第一步,观察数据范围。

我们发现 \(c\) 较小,\(n\) 也不算大,这启示我们可以设计一个由剩余油量和点构成的二维状态,又因为存在多组询问,我们无法接受时间复杂度为 \(O(nmT)\) 的 SPFA,所以考虑用 Dijkstra。

第二步,考虑状态转移。

在跑正常的最短路时,对于每一个已经更新完毕的点,我们都借助它来更新其他点,但在这题中,我们设计了二维的状态,在每一个点都有两种操作:扣油去往其他的点或是原地不动花钱加油,我们需要同时考虑这两种操作。

具体的说,对于每一个二维的点 \((x,m)\)(\(x\) 表示当前位置,\(m\) 表示当前油量),我们可以用这个点更新点 \((x,m+1)(m+1\le c)\) 和所有的 \((y,m-w_{xy})(y\in \text{to}_x)\),更新方法类似于 Dijkstra。

然后就可以轻松写出代码了!

代码

#include <bits/stdc++.h>
using namespace std;
const int N=2010,M=20020,C=105;//点数,边数,最大油箱容量 int to[M],nxt[M],head[N],w[M],p[N];//图,边权,点权
int idx,n,m,c,s,e,in1,in2,in3,ans,Q;
int vis[N][C],dis[N][C];//二维的点,vis表示是否访问,dis是距离 struct node{int x,cost,you;}now;
bool operator < (node a,node b){return a.cost>b.cost;}//在优先队列中按所花的钱排序
priority_queue <node> q; void add(int u,int v,int c){idx++;to[idx]=v;w[idx]=c;nxt[idx]=head[u];head[u]=idx;} int bfs(int s){
while(!q.empty()) q.pop();//一定要清空!!!
memset(vis,0,sizeof vis);//初始化
memset(dis,0x3f,sizeof dis);
q.push(node{s,0,0});
dis[s][0]=0;
while(!q.empty()){
now=q.top();q.pop();
if(now.x==e) return now.cost;//到达终点
if(vis[now.x][now.you]) continue;
vis[now.x][now.you]=1;
if(now.you<c) //如果当前油箱没有满
if(dis[now.x][now.you+1]>now.cost+p[now.x]){
dis[now.x][now.you+1]=now.cost+p[now.x];//更新状态
q.push(node{now.x,now.cost+p[now.x],now.you+1});//加油,入队
}
for(int i=head[now.x];i;i=nxt[i]){
if(now.you<w[i]) continue;//无法通过
if(dis[to[i]][now.you-w[i]]>now.cost){//可以通过,更新状态
dis[to[i]][now.you-w[i]]=now.cost;
q.push(node{to[i],now.cost,now.you-w[i]});//扣油,入队
}
}
}
return -1;
} int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d",&p[i]);
for(int i=1;i<=m;i++){
scanf("%d%d%d",&in1,&in2,&in3);
in1++;in2++;//点从0到n-1,我们统一加1
add(in1,in2,in3);add(in2,in1,in3);
}
scanf("%d",&Q);
while(Q--){
scanf("%d%d%d",&c,&s,&e);
s++;e++;
ans=bfs(s);
if(ans==-1) puts("impossible");//无解
else printf("%d\n",ans);
}
return 0;
}

Full Tank 题解的更多相关文章

  1. [题解](优先队列广搜)POJ_3635_Full Tank

    用二元组$(city,fuel)$即可记录所有状态,以当前花费为关键字优先队列,开数组记录直接做即可 有一个点在于每次不用枚举所有的加油数量,只需要加一即可,因为如果在加一升更优的话又会扩展出加更多油 ...

  2. POJ 3635 - Full Tank? - [最短路变形][手写二叉堆优化Dijkstra][配对堆优化Dijkstra]

    题目链接:http://poj.org/problem?id=3635 题意题解等均参考:POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]. 一些口胡: ...

  3. POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]

    题目链接:http://poj.org/problem?id=3635 Description After going through the receipts from your car trip ...

  4. leetcode & lintcode 题解

    刷题备忘录,for bug-free 招行面试题--求无序数组最长连续序列的长度,这里连续指的是值连续--间隔为1,并不是数值的位置连续 问题: 给出一个未排序的整数数组,找出最长的连续元素序列的长度 ...

  5. hdoj 4445 Crazy Tank 物理题/枚举角度1

    Crazy TankTime Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

  6. [LeetCode 题解]:Gas Station

    前言   [LeetCode 题解]系列传送门:  http://www.cnblogs.com/double-win/category/573499.html   1.题目描述 There are ...

  7. 【题解】APIO2013机器人

    其实这题前前后后的思考时间加起来应该有两天之久了,dp状态,转移方式等等都还是比较好想,然而左看右看觉得spfa复杂度未免太爆炸……然后选择看了一篇题解,发现在多重优化之下,其实是可以过的…… 首先建 ...

  8. UVA11367 Full Tank? 【分层图最短路】

    题目 After going through the receipts from your car trip through Europe this summer, you realised that ...

  9. Codeforces Round #436 (Div. 2) 题解864A 864B 864C 864D 864E 864F

    A. Fair Game time limit per test 1 second memory limit per test 256 megabytes input standard input o ...

  10. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

随机推荐

  1. C# - ConcurrentDictionary 并发场景使用注意事项

    1 自身作为 Enumerable 的遍历 自身作为可遍历对象,键值对为元素进行遍历,是线程安全的,但不提供快照,遍历过程中集合产生变更会直接反馈至此次遍历过程中.但并不一定能够保障获取数据的过程中, ...

  2. 上班第一天 Android 环境配置

    其实是昨天把大概 回归Android开发第一天 学会查 然后等待 反正我是不希望以后再查了 写出来吧 去谷歌那边把android studio下载下来 更新jdk版本(与传统的java开发不同 高版本 ...

  3. Centos7中Oracle占用CPU过高(解决方案)

    Centos7中Oracle占用CPU过高(解决方案) 前言: 99%的问题几乎都是SQL的问题,一般SQL可能会出现以下几种情况: 相关SQL搜索条件没有加索引 索引失效 联合查询过多 数据量过大 ...

  4. Cilium 系列-3-Cilium 的基本组件和重要概念

    系列文章 Cilium 系列文章 前言 安装完了,我们看看 Cilium 有哪些组件和重要概念. Cilium 组件 如上所述,安装 Cilium 时,会安装几个运行组件(有些是可选组件), 它们各是 ...

  5. 安装centos系统,硬盘检测报错:修改BIOS为 Legacy

    进bios,将模式修改为legacy. 硬盘使用 MBR 分区,需要用 Legacy BIOS 启动,而不是 UEFI BIOS . 至于为什么安装的时候会报错? 可能是系统有这方面限制.也可能是别的 ...

  6. iframe与主窗口通信

    1. 引言 <iframe> 元素是 HTML 中的一个标签,用于在当前页面中嵌入另一个页面 使用 <iframe> 可以实现以下功能: 嵌入其他网页:可以将其他网页嵌入到当前 ...

  7. 2021-7-8 VUE的简易表单提交

    <!DOCTYPE html> <html> <head> <title> </title> <style type="te ...

  8. SpringBoot里的Servlet和实现

    Servlet 接口,一个规范, SpringBoot Spring Boot 是 Spring 的子项目,正如其名字,提供 Spring 的引导( Boot )的功能. 通过 Spring Boot ...

  9. [golang]使用logrus自定义日志模块

    简介 logrus是一个第三方日志库,性能虽不如zap和zerolog,但方便易用灵活.logrus完全兼容标准的log库,还支持文本.JSON两种日志输出格式. 特点 相较于标准库,logrus有更 ...

  10. 使用 SQL 的方式查询消息队列数据以及踩坑指南

    背景 为了让业务团队可以更好的跟踪自己消息的生产和消费状态,需要一个类似于表格视图的消息列表,用户可以直观的看到发送的消息:同时点击详情后也能查到消息的整个轨迹. 消息列表 点击详情后查看轨迹 原理介 ...