【思路】

题目生词

figure

n. 数字

v. 认为,认定;计算;是……重要部分

The stations are represented by vertices and the roads correspond to the edges.

顶点表示车站,边表示道路。

correspond to 相当于

capacity

n. 能力;容量

题目大意:给出需要调整的车站编号,从0处出发,一路上顺便调整途径的车站,使得每个车站的车辆数是Cmax的一半,多的带走少的补齐。选最短路,最短相同选从0处带的车最少的路,若还相同则选择带回0处的车最少的路。

输出要带的车辆数,路径,带回的车辆数(返回时直接回,不再调整)。

方法:Dijkstra + DFS。先用Dijkstra算法算出最短路径,只考虑时间最短,建立vector保存路径的前驱节点。然后用DFS遍历每一条路径,获得一条路径后(即遍历到了起始节点0)计算带去带回的车辆,确定最佳方案。

计算带去、带回车辆数的方法:对于每个站点,考虑前一个站点传递下来的车辆数trans和自己的车辆数bike[i]相加的结果与cmax/2的差,分两种情况——一,差值为负,即车不够,要从0处带车,所以bring的值增加其差值的绝对值,而传递给下去的车辆数trans置为0;二,差值非负,表示车够了,多出的车赋值给trans传递到下一个站点,bring的值保持。一开始bring和trans的值为0,从出发节点的下一个节点(即0号节点的下一个节点)开始遍历计算。最终bring为要从0处携带的车辆数,trans即带回的车辆数。注意,有关从0处带多少车,只与当前走过的车站有关,即不管后面站点车再多,前面的车不够了,就要从0处带(因为走路不会回头,携带的车辆数是随着路径的推进而变化的)。而DFS是从终点向前推到起点结束,则必须要求完整条路径才能算的出来。最后倒着输出变长数组的值,即为路径。

【tips】要熟练运用Dijkstra+DFS求最佳路径的算法!

【AC代码】

 #define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<vector>
using namespace std;
#define N 502
#define INF 100000000
int cmax, stations, goal, roads;
int bike[N];
int length[N][N] = { {} };
bool vis[N] = {};
int d[N];
vector<int>pre[N];
vector<int>path;
int minbring = INF, minback = INF;
vector<int>bestpath;
void DFS(int v)
{
if (v == ) //到边界--起始节点。
{
//计算第二、三标尺的值,即带去、带回的车辆数。
path.push_back(v);
int bring = ;
int trans = ;
for (int i = path.size() - ; i >= ; i--)
{
int u = path[i];
if (bike[u] + trans >= cmax / )
{
trans += bike[u] - cmax / ;
}
else
{
bring += cmax / - bike[u] - trans;
trans = ;
}
}
//更新最优值。
if (bring < minbring)
{
bestpath = path;
minbring = bring;
minback = trans;
}
else if (bring == minbring)
{
if (trans < minback)
{
bestpath = path;
minback = trans;
}
}
path.pop_back();
return;
}
int i;
path.push_back(v);
for (i = ; i < pre[v].size(); i++)
{
DFS(pre[v][i]);
}
path.erase(path.end() - );//可以写成path.pop_back();
}
void Dijkstra(int s)
{
int i, j;
fill(d, d + N, INF);
d[s] = ;
for (i = ; i <= stations; i++)
{
//找不在s集中的d最小
int min = INF, u = -;
for (j = ; j <= stations; j++)
{
if (min > d[j] && vis[j] == )
{
min = d[j];
u = j;
}
}
if (u == -)return;
vis[u] = true;
//对于通过u能到s的点v,更新路径
for (j = ; j <= stations; j++)
{
if (length[u][j] && vis[j] == )
{
if (d[u] + length[u][j] < d[j])
{
d[j] = d[u] + length[u][j];
pre[j].clear();
pre[j].push_back(u);
}
else if (d[u] + length[u][j] == d[j])
{
pre[j].push_back(u);
}
}
} }
} int main()
{
cin >> cmax >> stations >> goal >> roads;
int i;
for (i = ; i <= stations; i++)
cin >> bike[i];
for (i = ; i < roads; i++)
{
int u, v;
cin >> u >> v;
cin >> length[u][v];
length[v][u] = length[u][v];
}
Dijkstra();
DFS(goal);
cout << minbring << "";
for (i = bestpath.size() - ; i >= ; i--)
{
cout << "->" << bestpath[i];
}
cout << " " << minback;
return ;
}

[PAT] A1018 Public Bike Management的更多相关文章

  1. PAT A1018 Public Bike Management (30 分)——最小路径,溯源,二标尺,DFS

    There is a public bike service in Hangzhou City which provides great convenience to the tourists fro ...

  2. PAT 1018 Public Bike Management[难]

    链接:https://www.nowcoder.com/questionTerminal/4b20ed271e864f06ab77a984e71c090f来源:牛客网PAT 1018  Public ...

  3. PAT甲级——A1018 Public Bike Management

    There is a public bike service in Hangzhou City which provides great convenience to the tourists fro ...

  4. PAT 1018. Public Bike Management

    There is a public bike service in Hangzhou City which provides great convenience to the tourists fro ...

  5. A1018. Public Bike Management

    There is a public bike service in Hangzhou City which provides great convenience to the tourists fro ...

  6. PAT 1018 Public Bike Management(Dijkstra 最短路)

    1018. Public Bike Management (30) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yu ...

  7. pat 甲级 Public Bike Management

    Public Bike Management (30) 题目描述 There is a public bike service in Hangzhou City which provides grea ...

  8. PAT_A1018#Public Bike Management

    Source: PAT A1018 Public Bike Management (30 分) Description: There is a public bike service in Hangz ...

  9. PAT甲级1018. Public Bike Management

    PAT甲级1018. Public Bike Management 题意: 杭州市有公共自行车服务,为世界各地的游客提供了极大的便利.人们可以在任何一个车站租一辆自行车,并将其送回城市的任何其他车站. ...

随机推荐

  1. rabbit MQ 消息队列

    为什么会需要消息队列(MQ)? 一.消息队列概述消息队列中间件是分布式系统中重要的组件,主要解决应用解耦,异步消息,流量削锋等问题,实现高性能,高可用,可伸缩和最终一致性架构.目前使用较多的消息队列有 ...

  2. 处理异常未知端口 Lsof命令

    需要使用的命令:  ss -tnl 显示所有tcp已被监听的端口  lsof -i:端口 显示所有打开该端口的进程 工作实例: 某天通过ss -tnl发现有不认识的正在被监听的端口 ? 于是使用lso ...

  3. [Redis-CentOS7]Python操作Redis(十一)

    Python 操作redis #!/usr/bin/env pyhton # coding:utf-8 # @Time : 2020-02-16 21:36 # @Author : LeoShi # ...

  4. React之虚拟DOM中的Diff算法

    一.React中的setState ( 异步函数,异步获取数据 ) 若操作的时间间隔短,它可以将多个setState结合成一个setState,减少虚拟DOM的比对次数,提高性能 二.同层虚拟DOM对 ...

  5. C++泛化单链表

    泛型单链表 单链表将每个数据分为节点,每个节点存储数据和指向下一个节点的指针.这样数据就不用在内存中使用连续的存储空间,有更大的灵活性. 这里将单链表分为节点类(Node)和链表类(singleLin ...

  6. IntelliJ 如何找到项目中 Deprecated 的方法

    在一个项目中,如果我们标记了某些元素为 Deprecated 的话,如何让我们能够快速找到? 简单来说,你可以对项目进行 Code Inspection. 选择 Analyze > Inspec ...

  7. iOS异常采用处理方式

    iOS开发过程中我们经常会遇到异常问题 对异常的处理一般采用打印或者直接抛出.这样可以很方便我们调试过程有所参考,而且方便我们查看异常产生的位置信息 NSError(错误信息) 采用NSError的情 ...

  8. centos docker redis 安装

    1.下载redis镜像 docker pull redis 2.下载redis.conf文件 https://redis.io/topics/config 这边查找自己服务器redis对应的版本文件 ...

  9. SparkShuffle机制

    在早期版本的Spark中,shuffle过程没有磁盘读写操作,是纯内存操作,后来发现效率较低,且极易引发OOME,较新版本的Shuffle操作都加入了磁盘读写进行了改进. 1.未经优化的HashShu ...

  10. mysql数据库技术1——基本的增删查改的sql语句

    1.数据库语言的分类 DDL:数据库定义语言 data Definition language 用于创建.修改.和删除数据库内的数据结构,如: 1:创建和删除数据库(CREATE DATABASE | ...