当前位置: 首页 > article >正文

用C语言搞定PTA数据结构7-1天梯地图:迪杰斯特拉算法实战与避坑指南

从零实现PTA天梯地图双权重迪杰斯特拉算法全解析当面对PTA数据结构7-1天梯地图这类双权重图的最短路径问题时许多初学者会陷入算法选择的困境。本文将彻底拆解如何用C语言实现这一经典题目不仅教你写出能AC的代码更重要的是掌握算法改造的思维方法。1. 问题重难点剖析这道题的核心在于处理带有两个权重距离和时间的有向图并需要满足以下特殊条件双最优路径输出需分别计算最快路径和最短距离路径次级条件处理当多条路径时间相同时选择其中距离最短的当多条路径距离相同时选择经过节点最少的路径还原需要记录并输出完整路径节点序列常见误区包括// 错误示例仅存储单一权重 typedef struct { int weight; // 无法同时表示时间和距离 // ... } Edge;2. 数据结构设计关键正确的数据结构设计是解题的基础。我们需要2.1 图的表示采用邻接矩阵存储双权重信息#define INF 0x3f3f3f3f typedef struct { int length; // 道路长度 int time; // 通行时间 } Edge; Edge graph[MAX_N][MAX_N]; // 邻接矩阵2.2 路径信息记录需要扩展传统迪杰斯特拉算法的存储结构typedef struct { int prev; // 前驱节点 int total_dist; // 累计距离 int total_time; // 累计时间 int node_count; // 节点计数用于距离相同时判断 } PathInfo;3. 算法改造实战标准迪杰斯特拉算法需要针对题目需求进行两处关键改造3.1 时间优先的最短路径void dijkstra_time(int start, int n, PathInfo time_path[]) { int visited[MAX_N] {0}; // 初始化 for (int i 0; i n; i) { time_path[i].total_time INF; time_path[i].total_dist INF; time_path[i].prev -1; time_path[i].node_count 0; } time_path[start].total_time 0; time_path[start].total_dist 0; time_path[start].node_count 1; for (int i 0; i n; i) { int u -1; // 找出当前未访问的最小时间节点 for (int j 0; j n; j) { if (!visited[j] (u -1 || time_path[j].total_time time_path[u].total_time)) u j; } if (time_path[u].total_time INF) break; visited[u] 1; // 松弛操作 for (int v 0; v n; v) { if (graph[u][v].time INF) continue; int new_time time_path[u].total_time graph[u][v].time; int new_dist time_path[u].total_dist graph[u][v].length; if (new_time time_path[v].total_time) { // 发现更优时间路径 time_path[v].total_time new_time; time_path[v].total_dist new_dist; time_path[v].prev u; time_path[v].node_count time_path[u].node_count 1; } else if (new_time time_path[v].total_time new_dist time_path[v].total_dist) { // 时间相同但距离更短 time_path[v].total_dist new_dist; time_path[v].prev u; time_path[v].node_count time_path[u].node_count 1; } } } }3.2 距离优先的最短路径void dijkstra_distance(int start, int n, PathInfo dist_path[]) { int visited[MAX_N] {0}; // 初始化与时间优先类似省略部分代码 // ... for (int i 0; i n; i) { int u -1; // 找出当前未访问的最小距离节点 for (int j 0; j n; j) { if (!visited[j] (u -1 || dist_path[j].total_dist dist_path[u].total_dist)) u j; } if (dist_path[u].total_dist INF) break; visited[u] 1; // 松弛操作 for (int v 0; v n; v) { if (graph[u][v].length INF) continue; int new_dist dist_path[u].total_dist graph[u][v].length; int new_time dist_path[u].total_time graph[u][v].time; int new_count dist_path[u].node_count 1; if (new_dist dist_path[v].total_dist) { // 发现更短距离路径 dist_path[v].total_dist new_dist; dist_path[v].total_time new_time; dist_path[v].prev u; dist_path[v].node_count new_count; } else if (new_dist dist_path[v].total_dist new_count dist_path[v].node_count) { // 距离相同但节点数更少 dist_path[v].total_time new_time; dist_path[v].prev u; dist_path[v].node_count new_count; } } } }4. 路径还原技巧获得最优路径后需要从终点回溯到起点void reconstruct_path(int end, PathInfo path[], int result[], int *count) { int current end; *count 0; // 反向存储路径 while (current ! -1) { result[(*count)] current; current path[current].prev; } // 反转路径数组 for (int i 0; i *count / 2; i) { int temp result[i]; result[i] result[*count - 1 - i]; result[*count - 1 - i] temp; } }5. 常见调试陷阱在实现过程中以下几个问题需要特别注意初始化问题// 错误示例忘记初始化node_count time_path[start].node_count 1; // 必须显式初始化边界条件处理// 处理没有路径的情况 if (path[end].total_time INF) { printf(No path exists\n); return; }输入数据读取// 读取道路信息时的常见错误 scanf(%d %d %d %d %d, v1, v2, one_way, len, time); if (one_way 0) { // 双向道路需要设置两个方向 graph[v2][v1].length len; graph[v2][v1].time time; }路径比较逻辑// 比较两条路径是否完全相同时的常见错误 int is_same_path 1; for (int i 0; i time_count; i) { if (time_path[i] ! dist_path[i] || time_count ! dist_count) { is_same_path 0; break; } }6. 性能优化建议当节点数N较大时接近题目上限500可以考虑以下优化优先队列优化// 使用最小堆替代线性搜索 typedef struct { int node; int time; int dist; } QueueItem; // 比较函数 int cmp_time(const void *a, const void *b) { return ((QueueItem*)a)-time - ((QueueItem*)b)-time; }稀疏图优化// 改用邻接表存储 typedef struct Node { int to; int length; int time; struct Node *next; } Node; Node *adj_list[MAX_N]; // 邻接表头指针数组提前终止// 当目标节点出队时提前终止 if (u target) break;7. 完整解决方案框架以下是整合各模块后的程序框架#include stdio.h #include string.h #define MAX_N 505 #define INF 0x3f3f3f3f // 数据结构定义省略 // 函数声明省略 int main() { int N, M; scanf(%d %d, N, M); // 初始化图 // ... // 读取道路信息 // ... int start, end; scanf(%d %d, start, end); PathInfo time_path[MAX_N], dist_path[MAX_N]; dijkstra_time(start, N, time_path); dijkstra_distance(start, N, dist_path); int time_route[MAX_N], dist_route[MAX_N]; int time_count 0, dist_count 0; reconstruct_path(end, time_path, time_route, time_count); reconstruct_path(end, dist_path, dist_route, dist_count); // 输出结果 // ... return 0; }在实际编码时建议先实现基本功能并通过样例测试再逐步添加次级条件判断最后进行边界条件测试和性能优化。这种分阶段实现的策略可以有效降低调试难度。

相关文章:

用C语言搞定PTA数据结构7-1天梯地图:迪杰斯特拉算法实战与避坑指南

从零实现PTA天梯地图:双权重迪杰斯特拉算法全解析 当面对PTA数据结构7-1天梯地图这类双权重图的最短路径问题时,许多初学者会陷入算法选择的困境。本文将彻底拆解如何用C语言实现这一经典题目,不仅教你写出能AC的代码,更重要的是掌…...

Proteus仿真进阶:用STM32F103驱动L298,深入理解PWM占空比与电机速度的映射关系

Proteus仿真进阶:用STM32F103驱动L298,深入理解PWM占空比与电机速度的映射关系 在嵌入式开发中,电机控制是一个经典且实用的课题。很多教程会告诉你如何通过STM32的PWM输出让电机转起来,但很少有人解释为什么代码中会出现"10…...

从‘打包’到‘压缩’:一文理清Linux tar命令的-z、-j、-J参数该怎么选(附性能对比)

从‘打包’到‘压缩’:一文理清Linux tar命令的-z、-j、-J参数该怎么选(附性能对比) 在Linux系统管理中,文件归档与压缩是每位开发者绕不开的基础操作。当你面对几十GB的日志文件需要备份,或是需要将数百张高分辨率图片…...

别再只用yum了!手把手教你用RPM包在CentOS 7.9上安装最新版LibreOffice 7.5.4(含中文包)

告别老旧版本:CentOS 7.9手动安装LibreOffice 7.5.4全攻略 在开源办公软件领域,LibreOffice无疑是当前最活跃、功能最全面的选择之一。然而许多CentOS用户发现,通过系统默认的yum仓库安装的LibreOffice版本往往落后官方最新版数年之久。以Cen…...

用STM32F103C8T6驱动Ra-01SC模组实现点对点通信(附完整代码与接线图)

STM32与Ra-01SC模组实战:从零搭建LoRa点对点通信系统 在物联网和远程监测领域,LoRa技术以其低功耗、远距离的特性成为无线通信的热门选择。Ra-01SC模组作为一款高性价比的LoRa模块,配合STM32F103C8T6这款经典MCU,能够快速构建稳定…...

SkyWalking UI 保姆级使用指南:从仪表盘到告警,手把手教你排查线上问题

SkyWalking UI 实战指南:从异常告警到代码级优化的全链路排查 当凌晨三点的告警短信突然亮起屏幕,作为值班工程师的你该如何快速定位线上服务的性能瓶颈?SkyWalking UI 提供的不仅是数据看板,更是一套完整的分布式系统诊断工具箱。…...

手把手教你用正点原子RV1126开发板玩转RKMedia:从录音到RTSP推流保姆级教程

手把手教你用正点原子RV1126开发板玩转RKMedia:从录音到RTSP推流保姆级教程 第一次拿到正点原子ATK-DLRV1126开发板时,那种既兴奋又忐忑的心情至今记忆犹新。作为一款基于Rockchip RV1126芯片的嵌入式开发平台,它强大的多媒体处理能力让人跃…...

KVM网络配置踩坑记:从virt-install的`--network`参数到virsh管理虚拟网桥

KVM网络配置实战:从virt-install到virsh的深度解析 当你在本地环境搭建KVM虚拟机时,网络配置往往是第一个拦路虎。不同于物理机插上网线就能用的简单体验,虚拟化环境中的网络需要经过多层抽象和配置才能正常工作。本文将带你深入KVM网络配置的…...

手把手教你用复旦微FM7Z045芯片在线调试DDR:JTAG与QSPI模式切换避坑指南

复旦微FM7Z045芯片DDR调试实战:模式切换与JTAG连接深度解析 第一次拿到复旦微FM7Z045开发板时,许多工程师都会遇到一个令人困惑的问题——明明按照手册步骤操作,DDR调试却总是失败。这往往不是代码问题,而是模式选择不当导致的。本…...

告别触摸屏开发烦恼:手把手教你用tslib 1.16搞定嵌入式Linux触摸校准与Qt适配

嵌入式Linux触摸屏开发实战:从tslib校准到Qt适配全解析 在工业控制、医疗设备和智能终端等嵌入式场景中,触摸屏作为最直接的人机交互方式,其精度和响应速度直接影响用户体验。然而在实际开发中,工程师们常会遇到触摸坐标漂移、点击…...

从投稿到录用:我是如何用IEEE官方Word模板搞定格式,让审稿人一眼舒服的?

从投稿到录用:我是如何用IEEE官方Word模板搞定格式,让审稿人一眼舒服的? 第一次投稿IEEE期刊时,我花了整整三天时间调整格式——页眉页脚错位、参考文献编号混乱、图表标题忽大忽小。直到收到编辑的退修邮件:"请…...

别急着换件!汇川伺服报Er.136/Er.740编码器故障,先按这3步自查(附线缆选购建议)

汇川伺服编码器故障排查指南:从干扰溯源到线缆优化 工业现场最让人头疼的莫过于设备间歇性抽风——明明昨天还运行良好,今天却频繁报Er.136或Er.740编码器故障。作为经历过数十次类似案例的技术老兵,我必须强调:80%的编码器问题根…...

智慧树自动刷课插件:3分钟安装的终极学习效率提升指南

智慧树自动刷课插件:3分钟安装的终极学习效率提升指南 【免费下载链接】zhihuishu 智慧树刷课插件,自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 还在为智慧树平台的冗长视频课程烦恼吗?智…...

告别快捷键混乱!PowerToys保姆级教程:让Win键位秒变Mac,开发效率翻倍

告别快捷键混乱!PowerToys保姆级教程:让Win键位秒变Mac,开发效率翻倍 作为一名长期在Windows和Mac双平台切换的开发者,最令人抓狂的莫过于快捷键的差异。每次从Mac切换到Windows,肌肉记忆总会在关键时刻背叛你——当你…...

N5105 4口2.5g V3 Intel i225 PVE 6.2下的Openclaw安装

一、Ubuntu 26.04安装 1. 从官网上下载ubuntu 26.04 LTS版本 下载地址:Download Ubuntu Desktop | Ubuntu 2. 将下载好的iso文件上传到pve中,登录PVE后台,点击local->ISO镜像->上传 3. 创建虚拟机 其他按默认配置即可。 4. 安装Ubu…...

DeepSeek LeetCode 2508.添加边使所有节点度数都为偶数 public boolean isPossible(int n, List<List<Integer>> edges)

问题分析我们需要判断能否添加至多两条边(不能添加重复边,不能添加自环),使得图中所有节点的度数都为偶数。---思路步骤1. 统计每个节点的当前度数遍历给出的边,统计每个节点的度数。 2. 找出度数为奇数的节点设奇数度…...

30天无限续杯:JetBrains IDE评估重置神器全攻略

30天无限续杯:JetBrains IDE评估重置神器全攻略 【免费下载链接】ide-eval-resetter 项目地址: https://gitcode.com/gh_mirrors/id/ide-eval-resetter 你是否曾经在深夜coding时,突然被IDE弹出的"试用期已结束"提示打断思路&#xff…...

3分钟快速上手:Hanime1Plugin安卓插件打造纯净动画观影体验终极指南

3分钟快速上手:Hanime1Plugin安卓插件打造纯净动画观影体验终极指南 【免费下载链接】Hanime1Plugin Android插件(https://hanime1.me) (NSFW) 项目地址: https://gitcode.com/gh_mirrors/ha/Hanime1Plugin 你是否厌倦了动画观影时被各种广告弹窗打断&#x…...

被AI冲击的App,反成了Agent的命门

2026年最流行的一个判断:AI Agent要吃掉一切图形界面,对话即服务,App即将消亡。 这个判断的依据并非没有道理。Agent确实在接管"发现"和"调度"——用户不再需要主动打开某个App,而是告诉Agent"帮我订一…...

VSCode+GCC+OpenOCD:打造你的STM32专属OpenHarmony 3.1开发流水线

VSCodeGCCOpenOCD:构建STM32 OpenHarmony开发的高效流水线 在嵌入式开发领域,效率往往取决于工具链的整合程度。当OpenHarmony遇上STM32,如何摆脱传统IDE的束缚,打造一套现代化、可定制的开发环境?本文将带你从零搭建基…...

从SDF反标失败说起:为什么PBA模式的结果不能写进标准延迟文件?

从SDF反标失败看PBA与GBA的本质差异:芯片设计中的精度与效率博弈 当你在PrimeTime中完成了一次精细的PBA模式时序分析,确认设计满足所有时序约束后,尝试将结果导出为SDF文件用于后仿验证时,工具却报错或生成的SDF文件无法正确反映…...

猫抓Cat-Catch:浏览器资源嗅探神器,轻松下载网页视频和流媒体资源

猫抓Cat-Catch:浏览器资源嗅探神器,轻松下载网页视频和流媒体资源 【免费下载链接】cat-catch 猫抓 浏览器资源嗅探扩展 / cat-catch Browser Resource Sniffing Extension 项目地址: https://gitcode.com/GitHub_Trending/ca/cat-catch 你是否曾…...

母线槽核心部件解析 —— 高纯铜导体与绝缘层的技术价值

在低压配电系统中,母线槽凭借大电流传输能力、高安全性及长寿命特性,成为大型基建、工业厂房、商业建筑等场景的核心配电设备。 扬中金展电气深耕母线槽研发生产 16 年,以严苛的材质标准与精密工艺,打造高可靠母线槽产品&#xff…...

【职场】职场里,“被喜欢“和“被重用“是两件完全不同的事

职场里,"被喜欢"和"被重用"是两件完全不同的事我见过太多这样的人。 在公司里人缘极好,谁都说他靠谱,谁都愿意跟他合作。 开会时第一个帮人倒水,群里消息第一个回复,同事生日永远记得,…...

【求职】衡量你职场流通性的,从来不是你的能力

衡量你职场流通性的,从来不是你的能力先问你一个问题。 你上一次被猎头主动联系,是什么时候? 如果你需要认真回忆,那这篇文章,你需要认真读完。一、"流通性"是个被严重低估的职场变量 大多数人谈职业发展&am…...

【职场】为什么越努力的人,在职场死得越惨?

为什么越努力的人,在职场死得越惨? ——没有人告诉你,努力本身是一种暴露。一、先说一个你亲眼见过,但从没想明白的现象 你身边一定有这样的人: 工作最拼的那个,最后被裁了。 加班最多的那个,升…...

链路层协议

链路层协议要解决哪些问题。有哪些二层网络,其链路层协议是什么 链路层(数据链路层,OSI模型第二层)的主要功能是在物理层提供的物理连接基础上,提供可靠的数据传输服务。它负责将原始的物理连接转化为无差错、有逻辑结…...

终极IDE评估周期管理方案:开源ide-eval-resetter完整解析

终极IDE评估周期管理方案:开源ide-eval-resetter完整解析 【免费下载链接】ide-eval-resetter 项目地址: https://gitcode.com/gh_mirrors/id/ide-eval-resetter 在当今快节奏的开发环境中,JetBrains IDE系列产品凭借其卓越的代码智能和丰富的功…...

技术分享 | 彻底解决图片“躺平”问题:Java 后端强制校准图片方向

在日常开发中,你是否遇到过这样的情况:前端上传了一张手机拍摄的照片,预览时明明是正的,存入服务器后却莫名其妙地“躺平”了,或者逆时针旋转了 90 度?以下方案用于强制旋转图片这通常是因为 JPEG 图片的 E…...

手把手教你用Google Cloud语音API为Android App加个“耳朵”和“嘴巴”(附免费额度避坑指南)

实战指南:在Android应用中集成Google Cloud语音技术 想象一下,你的Android应用能够听懂用户说话,还能用自然流畅的语音回应——这不再是科幻电影里的场景。借助Google Cloud的语音API,即使是独立开发者也能快速为应用添加专业的语…...