信息学奥赛一本通 2101:【23CSPJ普及组】旅游巴士(bus) | 洛谷 P9751 [CSP-J 2023] 旅游巴士

【题目链接】

ybt 2101:【23CSPJ普及组】旅游巴士(bus)
洛谷 P9751 [CSP-J 2023] 旅游巴士

【题目考点】

1. 图论:求最短路Dijkstra, SPFA
2. 动态规划
3. 二分答案
4. 图论:广搜BFS

【解题思路】

解法1:Dijkstra堆优化

每个地点是一个顶点,每条道路是一条边,道路只能单向通行,该图是有向图。通过每条边用时都是1单位时间,那么该图是无权图。每条道路都有开放时刻a,也就是说对于每条边:该边的a时刻后,才存在,在a时刻前不存在。
该题问的是离开景区的最早时间,也就是到达顶点n的最早时间。
如果给定进入园区的时刻,即便每条边有开放时间的限制,我们依然可以通过求最短路算法(如BFS,Dijkstra,SPFA)得到从起点到终点的最短路径长度。但到达终点的时刻未必是k的倍数。
观察变量取值范围,看到 1 ≤ k ≤ 100 1\le k \le 100 1k100,这是本题的突破点。
要想使到达顶点n的时刻是k的倍数,那么到达前一个顶点的时刻一定应该满足模k等于k-1(除以k的余数是k-1),到达路径上再前一个顶点的时刻应该满足模k等于k-2。。。
因此到达某顶点的时刻模k的值也是我们的需要考虑的限制。

状态定义:
  • 阶段:到达顶点i,到达该顶点的时间模k的值
  • 决策:下一步走到哪个邻接点
  • 策略:路径
  • 策略集合:在k的整数倍时刻从起点出发到达顶点i的所有路径
  • 条件:时刻最早
  • 统计量:时刻

d p [ i ] [ j ] dp[i][j] dp[i][j]:在k的整数倍时刻从起点出发到达顶点i,到达顶点i的时刻模k等于j的最早时刻。

状态转移方程:
  • 策略集合:在k的整数倍时刻从起点出发到达顶点i的所有路径
  • 分割策略集合:根据到达顶点v的前一个顶点的情况分割策略集合

由于本题没有说明整个图是有向无环图,因此不能使用拓扑排序的方法。
每个状态都可能多次被更新,因此需要使用Dijkstra或SPFA算法,更新状态变量。
可以从顶点u走到下一个顶点v时,更新顶点v的状态。
假设在第0时刻从顶点出发,在 t t t时刻到达顶点u,边<u,v>的开放时间为 a a a

  • 如果 t ≥ a t\ge a ta,通过顶点u到达顶点v的最早时刻 t m i n = t + 1 tmin = t+1 tmin=t+1
  • 如果 t < a t<a t<a,为了从顶点u通过边<u,v>走到顶点v,需要在入园前等待一段时间后再入园,入园时刻必须是 k k k的整数倍。(这个人在游玩时不能在任何位置停留,那么只能在入园前等待)
    如果等待 k k k时间后再入园,走相同路径到达顶点u时的时刻就是 t + k t+k t+k
    如果等待 2 k 2k 2k时间后再入园,走相同路径到达顶点u时的时刻就是 t + 2 k t+2k t+2k
    如果等待 x k xk xk时间后再入园,走相同路径到达顶点u时的时刻就是 t + x k t+xk t+xk。希望到达顶点u时边<u,v>已开放,则需要满足不等式
    t + x k ≥ a t+xk\ge a t+xka,即 x ≥ a − t k x \ge \frac{a-t}{k} xkat x x x最小为 ⌈ a − t k ⌉ \lceil \frac{a-t}{k} \rceil kat
    因此如果想要通过<u,v>到达顶点v,那么到达顶点u的最早时刻为 t + ⌈ a − t k ⌉ k t+\lceil \frac{a-t}{k} \rceil k t+katk
    因此如果 t < a t<a t<a,通过顶点u到达顶点v的最早时刻为 t + ⌈ a − t k ⌉ k + 1 t+\lceil \frac{a-t}{k} \rceil k+1 t+katk+1

因此存在一条到达顶点v的最早到达时刻为 t m i n tmin tmin的路径,使用 t m i n tmin tmin更新 d p [ v ] [ t m i n % k ] dp[v][tmin\%k] dp[v][tmin%k]。即如果 d p [ v ] [ t m i n % k ] > t m i n dp[v][tmin\%k]>tmin dp[v][tmin%k]>tmin,那么设 d p [ v ] [ t m i n % k ] = t m i n dp[v][tmin\%k]=tmin dp[v][tmin%k]=tmin

具体实现

如果到达某个顶点的最早时刻发生更新,而后应该再更新到达其邻接点的最早时刻。
该过程可以使用Dijkstra堆优化算法完成,需要把达顶点v的最早到达时刻为 t m i n tmin tmin的路径加入到优先队列。
到达顶点n,最早到达时刻模k等于0的最早到达时刻 d p [ n ] [ 0 ] dp[n][0] dp[n][0]就是最终结果。

复杂度分析

每个顶点有k个状态,因此可以将每个顶点看作k个顶点,原来的每条边可以看作k条边,整个图变为有nk个顶点,mk条边的图。
已知Dijkstra堆优化的时间复杂度为 O ( m k ⋅ l o g ( m k ) ) O(mk\cdot log(mk)) O(mklog(mk)),m最大是 2 ∗ 1 0 4 2*10^4 2104,k最大是100,该复杂度可以接受。

解法2:二分答案+bfs

可以反向思考,先通过二分答案确定到达终点n的时刻 t e te te,保证te是k的整数倍。
由于 a ≤ 1 0 6 a\le 10^6 a106,当开始时刻达到 1 0 6 10^6 106时,图中所有边都开放了。
边数 m ≤ 2 ∗ 1 0 4 m\le 2*10^4 m2104,假设在大约 1 0 6 10^6 106时刻出发,从顶点1到顶点n的路径需要经过所有的边,到达时刻不超过 1 0 7 10^7 107
由于k最大为100,我们可以二分到达时刻除以k的值,该值最小为0,最大值设大一点,就写 1 0 7 10^7 107,这样求出的 t e te te最小为0,最大不超过 1 0 9 10^9 109
判断解 t e te te是否满足条件:
建立原图的反图。从顶点n开始进行广搜。
如果搜索到顶点u时的时刻为t,u有邻接点v,在原图中,就是看在t-1时刻边<v,u>是否已开放,已知<v,u>的开放时间是a,如果 t − 1 ≥ a t-1\ge a t1a,那么可以在原图中从顶点v经过<v,u>到达顶点u。
在反图中,也就是判断如果 t − 1 ≥ a t-1\ge a t1a,那么边<u,v>已开放,可以在反图中从顶点u访问到邻接点v。
看是否存在到达顶点1的,到达时刻是k的整数倍的路径。
设vis数组,需要同时考虑到达顶点i,以及到达该顶点的时刻模k的值。
vis[i][j]表示到达顶点i,且到达时刻模k等于j的情况是否已发生。

因为在该问题的广搜的过程中,到达顶点的时刻是不断减少的,如果已经发生过到达顶点i且时刻模k等于j的情况,设该时刻是 t 1 t_1 t1。再次发生到达顶点i且时刻模k等于j时的时刻是 t 2 t_2 t2,则一定有 t 1 > t 2 t_1>t_2 t1>t2
如果从顶点i,时刻 t 1 t_1 t1出发通过一条路径到达顶点1,到达顶点1的时刻模k不等于0。那么从顶点i,时刻 t 2 t_2 t2出发经过相同的路径到达顶点1,到达顶点1的时刻模k的值和上面的情况中的值一定是一样的,也不为0。
因此如果已经发生过到达顶点i,到达时刻模k等于j的情况,就不用考虑后面出现的到达顶点i,到达时刻模k等于j的情况。

在反图进行广搜的过程中,对于每个出队的顶点u及到达u的时刻t
遍历u的邻接点v,到达顶点v的时刻为 t − 1 t-1 t1,<u,v>的开放时间为a

  • 如果 t − 1 < a t-1<a t1<a,则在原图中到达v时<v,u>边未开放,略过这种情况。
  • 如果到达顶点v,到达时刻模k等于 ( t − 1 ) % k (t-1)\%k (t1)%k的情况已经发生过,则略过。

不满足以上情况,才要访问顶点v:
如果顶点v就是顶点1,且到达的时刻 t − 1 t-1 t1是k的倍数,则找到了一个可行的解,答案出园时刻 t e te te满足条件。
而后存在到达顶点v,时刻为 t − 1 t-1 t1的情况,将该情况加入队列。并标记到达顶点v,时刻模k等于 ( t − 1 ) % k (t-1)\%k (t1)%k的情况已存在。
如果广搜结束后也没有找到可行的解,则出园时刻 t e te te不满足条件,返回假。

【题解代码】

解法1:Dijkstra堆优化算法+动态规划

#include<bits/stdc++.h>
using namespace std;
const int N = 10005, K = 100, INF = 0x3f3f3f3f;
struct Path
{int u, t;//存在到达顶点u,时刻为t的路径 bool operator < (const Path &b) const{return b.t < t;//时刻t更小更优先 }
};
struct Edge
{int v, a;
};
int n, m, k, dp[N][K];//在k的整数倍时刻从起点出发到达顶点i,到达顶点i的时刻模k等于j的最早时刻。
vector<Edge> edge[N];
int divCeil(int a, int b)//ceil(a/b)
{return (a-1)/b+1; 
} 
void dijkstra()
{priority_queue<Path> pq;memset(dp, 0x3f, sizeof(dp));//求最短路径,先将状态设为无穷 dp[1][0] = 0;//到达顶点1(起点),到达时刻模k为0的最早时刻为0pq.push(Path{1, 0});while(!pq.empty()){int u = pq.top().u, t = pq.top().t;pq.pop();for(Edge e : edge[u]){int v = e.v, a = e.a, tmin;//tmin:想要从u通过<u,v>,到达v的最早时刻 if(t >= a)tmin = t+1;elsetmin = t+1+divCeil(a-t, k)*k;if(dp[v][tmin%k] > tmin){dp[v][tmin%k] = tmin;pq.push(Path{v, tmin});}}}
}
int main()
{int u, v, a;cin >> n >> m >> k;for(int i = 1; i <= m; ++i){cin >> u >> v >> a;edge[u].push_back(Edge{v, a});}dijkstra();cout << (dp[n][0] == INF ? -1 : dp[n][0]);return 0;
}
解法2:二分答案+bfs
#include<bits/stdc++.h>
using namespace std;
const int N = 10005, K = 100, INF = 0x3f3f3f3f;
struct Path
{int u, t;//到达顶点u,到达时刻t 
};
struct Edge
{int v, a;
};
int n, m, k;
vector<Edge> rg[N];//反图 
bool vis[N][K];
bool check(int te)//判断到达时刻为te时,是否存在从n到1的,到达顶点1的时刻模k等于0的路径。 
{memset(vis, 0, sizeof(vis));queue<Path> que;vis[n][te%k] = true;que.push(Path{n, te});while(!que.empty()){int u = que.front().u, t = que.front().t;que.pop();for(Edge e : rg[u]){int v = e.v, a = e.a;if(t-1 >= a && !vis[v][(t-1)%k])//到顶点v时<v,u>已开放,同时现在没有在访问时刻模k等于(t-1)%k时访问顶点v {if(v == 1 && (t-1)%k == 0)//找到解,te满足条件 return true;vis[v][(t-1)%k] = true;que.push(Path{v, t-1});}}}return false;//te不满足条件 
}
int main()
{int u, v, a;cin >> n >> m >> k;for(int i = 1; i <= m; ++i){cin >> u >> v >> a;rg[v].push_back(Edge{u, a});//建反图}int l = 0, r = 1e7; while(l < r){int mid = (l+r)/2;if(check(mid*k))//答案是mid*k,mid*k最大为1e9r = mid;elsel = mid+1; }if(check(l*k))//二分得到的答案也未必是满足条件的,需要再判断一下 cout << l*k;elsecout << -1;return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/69246.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C基础寒假练习(6)

一、终端输入行数&#xff0c;打印倒金字塔 #include <stdio.h> int main() {int rows;printf("请输入倒金字塔的行数: ");scanf("%d", &rows);for (int i rows; i > 0; i--) {// 打印空格for (int j 0; j < rows - i; j) {printf(&qu…

vim modeline

1. 什么是 Vim 模型行&#xff08;modeline&#xff09;&#xff1f; Vim 模型行是嵌入在文件中的特殊注释行&#xff0c;用于告诉 Vim 编辑器如何配置编辑选项。它的语法格式如下&#xff1a; # vim: 选项1值1:选项2值2:...它以 # vim: 开头&#xff08;# 是注释符&#xff…

【C# 】图像资源的使用

在C#中&#xff0c;图像资源的使用方式方法主要依赖于你所使用的框架和库。以下是几种常见的使用图像资源的方法&#xff1a; Windows Forms 直接加载图像&#xff1a; 使用System.Drawing.Image.FromFile()方法可以直接从文件系统加载图像。 Image image Image.FromFile(&qu…

OpenGL学习笔记(六):Transformations 变换(变换矩阵、坐标系统、GLM库应用)

文章目录 向量变换使用GLM变换&#xff08;缩放、旋转、位移&#xff09;将变换矩阵传递给着色器坐标系统与MVP矩阵三维变换绘制3D立方体 & 深度测试&#xff08;Z-buffer&#xff09;练习1——更多立方体 现在我们已经知道了如何创建一个物体、着色、加入纹理。但它们都还…

java后端开发面试常问

面试常问问题 1 spring相关 &#xff08;1&#xff09;Transactional失效的场景 <1> Transactional注解默认只会回滚运行时异常&#xff08;RuntimeException&#xff09;&#xff0c;如果方法中抛出了其他异常&#xff0c;则事务不会回滚&#xff08;数据库数据仍然插…

使用conda创建自己的python虚拟环境,与其他python版本独立区分

使用 Conda 创建和使用自己的运行环境非常简单&#xff0c;以下是详细步骤&#xff1a; 1. 安装 Anaconda 或 Miniconda 如果你尚未安装 Anaconda 或 Miniconda&#xff0c;可以访问 Anaconda 官网 或 Miniconda 官网 下载并安装。 2. 创建新的 Conda 虚拟环境 创建虚拟环境…

OSPF基础(1):工作过程、状态机、更新

OSPF基础 1、技术背景&#xff08;与RIP密不可分&#xff0c;因为RIP中存在的问题&#xff09; RIP中存在最大跳数为15的限制&#xff0c;不能适应大规模组网周期性发送全部路由信息&#xff0c;占用大量的带宽资源以路由收敛速度慢以跳数作为度量值存在路由环路可能性每隔30秒…

python爬虫--简单登录

1&#xff0c;使用flask框架搭建一个简易网站 后端代码app.py from flask import Flask, render_template, request, redirect, url_for, sessionapp Flask(__name__) app.secret_key 123456789 # 用于加密会话数据# 模拟用户数据库 users {user1: {password: password1}…

机器学习模型--线性回归、逻辑回归、分类

一、线性回归 级别1&#xff1a;简单一元线性回归&#xff08;手工实现&#xff09; import numpy as np import matplotlib.pyplot as plt# 生成数据 X np.array([1, 2, 3, 4, 5]) y np.array([2, 4, 5, 4, 5])# 手动实现梯度下降 def gradient_descent(X, y, lr0.01, epo…

ASP.NET Core与EF Core的集成

目录 分层项目中EF Core的用法 数据库的配置 数据库迁移 步骤汇总 注意&#xff1a; 批量注册上下文 分层项目中EF Core的用法 创建一个.NET类库项目BooksEFCore&#xff0c;放实体等类。NuGet&#xff1a;Microsoft.EntityFrameworkCore.RelationalBooksEFCore中增加实…

如何在React中使用Redux进行状态管理?

在现代前端开发中&#xff0c;React已成为构建用户界面的流行选择。然而&#xff0c;随着应用规模的不断增长&#xff0c;管理组件之间的状态变得愈加复杂。为了解决这一问题&#xff0c;Redux 作为一种状态管理工具应运而生。本文将详细介绍如何在React中集成和使用Redux来进行…

HTML中的图片标签详解及路径使用【学术投稿-第五届环境资源与能源工程国际学术会议(ICEREE 2025)】

官网&#xff1a;www.iceree.org 会议时间&#xff1a;2025年2月21-23日 会议地点&#xff1a;中国-昆明 简介 第五届环境资源与能源工程国际学术会议&#xff08;ICEREE 2025&#xff09;将于2025年2月21日至23日在中国昆明隆重举行。主要围绕“能源工程和能源技术”、“环…

react的antd表格自定义图标

将原版的加号换成箭头 自定义图标 安装图标包&#xff1a; npm install --save ant-design/icons 引入&#xff1a; import { RightOutlined, DownOutlined } from ant-design/icons; 参数是一个函数 <Table columns{columns} dataSource{data} indentSize{20}expandIc…

【回溯+剪枝】单词搜索,你能用递归解决吗?

文章目录 79. 单词搜索解题思路&#xff1a;回溯&#xff08;深搜&#xff09; 剪枝 79. 单词搜索 79. 单词搜索 ​ 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 …

Redis企业开发实战(二)——点评项目之商户缓存查询

目录 一、缓存介绍 二、缓存更新策略 三、如何保证redis与数据库一致性 1.解决方案概述 2.双写策略 3.双删策略 3.1延迟双删的目的 4.数据重要程度划分 四、缓存穿透 (一)缓存穿透解决方案 (二)缓存穿透示意图 五、缓存雪崩 (一)缓存雪崩解决方案 (二)缓存雪崩…

【C语言】常量指针和指针常量,指针数组和数组指针,指针函数和函数指针怎么区分?

文章目录 1、常量指针和指针常量2、指针数组和数组指针3、指针函数和函数指针 1、常量指针和指针常量 int a 0; const int* p &a;//常量指针 int* const p1 &a;//指针常量常量指针&#xff1a;无法通过该指针修改指向地址中的内容。 指针常量&#xff1a;指针指向的…

maven如何不把依赖的jar打包到同一个jar?

spring boot项目打jar包部署&#xff1a; 经过以下步骤&#xff0c; 最终会形成maven依赖的多个jar&#xff08;包括lib下添加的&#xff09;、 我们编写的程序代码打成一个jar&#xff0c;将程序jar与 依赖jar分开&#xff0c;便于管理&#xff1a; success&#xff1a; 最终…

Mac 部署Ollama + OpenWebUI完全指南

文章目录 &#x1f4bb; 环境说明&#x1f6e0;️ Ollama安装配置1. 安装[Ollama](https://github.com/ollama/ollama)2. 启动Ollama3. 模型存储位置4. 配置 Ollama &#x1f310; OpenWebUI部署1. 安装Docker2. 部署[OpenWebUI](https://www.openwebui.com/)&#xff08;可视化…

leetcode——爬楼梯(java)

假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢&#xff1f; 示例 1&#xff1a; 输入&#xff1a;n 2 输出&#xff1a;2 解释&#xff1a;有两种方法可以爬到楼顶。 1. 1 阶 1 阶 2. 2 阶 示例 2&#x…

在C#中,Array,List,ArrayList,Dictionary,Hashtable,SortList,Stack的区别

Array Array你可以理解为是所有数组的大哥 普通数组 : 特点是长度固定, 只能存储相同类型的数据 static void Main(string[] args){//声明int[] ints;string[] strings;People[] peoples;//默认值 //int 类型是 0//string 类型是 nullint[] ints1 { 1, 2, 3 };string[] …