P9751 [CSP-J 2023] 旅游巴士

P 9751 P9751 P9751

部分分思路

题目要求时间必须是 k k k 的非负整数倍,所以想到了升维 。这样就变成了一道分层图最短路 的题目。用 BFS 算法可以拿到 A i = 0 A_i=0 Ai=0 的 35 35 35 分。

满分思路

其实部分分的思路已经很接近正解了,想要拿到满分只需要做一点小小的调整。虽然说不能在路上停留,但是我们可以晚一点到达起点 。但是要注意:到达起点的时间也必须是 k k k 的倍数。这个做法 BFS 就解决不了了(它只能解决出发时间相同且边权为 1 1 1 的最短路问题),我们可以使用 Dijkstra 算法来解决这道题。时间复杂度约 O ( O( O( n n n + + + m ⋅ l o g 2 m m \cdot log_2m m⋅log2m ) ) )。

代码

cpp 复制代码
#include <cstdio>
#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
#include <cstring>
#include <cmath>
using namespace std;
const int INF = 0x3f3f3f3f; // 极大值∞

int n, m, k;
int dis[10010][110]; // 最短路
int vis[10010][110]; // 记录点有没有被选过

struct edge // 边
{
	int y, w;
} ;

struct node // 优先队列中的点
{
	int x, t, d;
	
	bool operator < (const node b) const // 重载运算符
	{
		return d > b.d;
	}
} ;

vector<edge> g[10010]; // 图

void add(int x, int y, int w) // 建边
{
	g[x].push_back({y, w});
}

void dijkstra(int s) // dijkstra算法,堆优化
{
	priority_queue<node> q;
	memset(dis, 0x3f, sizeof(dis));
	q.push({s, 0, 0});
	dis[s][0] = 0;
	while (q.size())
	{
		int x = q.top().x;
		int t = q.top().t;
		q.pop();
		if (vis[x][t])
			continue;
		vis[x][t] = 1;
		int nt = (t + 1) % k;
		for (int i = 0; i < g[x].size(); i++)
		{
			int y = g[x][i].y;
			int w = g[x][i].w;
			int d = dis[x][t];
			if (d < w) d += (w - d + k - 1) / k * k; // 到达起点时间
			if (dis[y][nt] > d + 1)
			{
				dis[y][nt] = d + 1;
				q.push({y, nt, dis[y][nt]});
			}
		}
	}
}

int main()
{
	cin >> n >> m >> k;
	for (int i = 1; i <= m; i++)
	{
		int u, v, w;
		cin >> u >> v >> w;
		add(u, v, w); // 建条单向边
	}
	dijkstra(1);
	if (dis[n][0] == INF)
		cout << "-1" << endl; // 无解
	else cout << dis[n][0] << endl;
	return 0;
}
相关推荐
sjsjs1126 分钟前
【动态规划-最长公共子序列(LCS)】力扣97. 交错字符串
算法·leetcode·动态规划
福楠39 分钟前
[LeetCode] 143. 重排链表
数据结构·c++·算法·leetcode·链表
无夜_39 分钟前
Builder(建造者模式)
开发语言·c++
大二转专业41 分钟前
408算法题leetcode--第29天
考研·算法·leetcode
梦城忆1 小时前
抽象类、接口和异常处理
java·开发语言·算法
Rookie也要加油1 小时前
06_23 种设计模式之《适配器模式》
c++·设计模式·适配器模式
zeroner_1 小时前
【C++】空指针和野指针
开发语言·c++
言存1 小时前
链表Set_LinkList(并集)
算法·链表
luthane1 小时前
python 实现graph list图列算法
python·算法·list
Rookie也要加油1 小时前
05_23 种设计模式之《建造者模式》
c++·设计模式·建造者模式