2024年信奥赛C++提高组csp-s初赛真题及答案解析(完善程序第2题)

2024年信奥赛C++提高组csp-s初赛真题及答案解析(完善程序第2题)

第 2 题

(次短路) 已知一个有 n个点 m条边的有向图 G**,并且给定图中的两个点 s 和 t,求次短路(长度严格大于最短路的最短路径)。如果不存在,输出一行 −1。如果存在,输出两行,第一行表示次短路的长度,第二行表示次短路的一个方案。

cpp 复制代码
#include <cstdio>
#include <queue>
#include <utility>
#include <cstring>
using namespace std;
 
const int maxn = 2e5+10, maxm = 1e6+10, inf = 522133279;
 
int n, m, s, t;
int head[maxn], nxt[maxm], to[maxm], w[maxm], tot = 1;
int dis[maxn<<1], *dis2;
int pre[maxn<<1], *pre2;
bool vis[maxn<<1];
 
void add(int a, int b, int c) {
    ++tot;
    nxt[tot] = head[a];
    to[tot] = b;
    w[tot] = c;
    head[a] = tot;
}
 
bool upd(int a, int b, int d, priority_queue<pair<int, int>> &q) {
    if (d >= dis[b]) return false;
    if (b < n) ___①___;
    q.push(___②___);
    dis[b] = d;
    pre[b] = a;
    return true;
}
 
void solve() {
    priority_queue<pair<int, int>> q;
    q.push(make_pair(0, s));
    memset(dis, ___③___, sizeof(dis));
    memset(pre, -1, sizeof(pre));
    dis2 = dis+n;
    pre2 = pre+n;
    dis[s] = 0;
    while (!q.empty()) {
        int aa = q.top().second; q.pop();
        if (vis[aa]) continue;
        vis[aa] = true;
        int a = aa % n;
        for (int e = head[a]; e; e = nxt[e]) {
            int b = to[e], c = w[e];
            if (aa < n) {
                if (!upd(a, b, dis[a]+c, q))
                    ___④__;
            } else {
                upd(n+a, n+b, dis2[a]+c, q);
        }
    }
}
 
void out(int a) {
    if (a != s) {
        if (a < n) out(pre[a]);
        else out(___⑤___);
    }
    printf("%d%c", a%n+1, " \n"[a == n+t]);
}
 
int main() {
    scanf("%d%d%d%d", &n, &m, &s, &t);
    s--, t--;
    for (int i = 0; i < m; ++i) {
        int a, b, c;
        scanf("%d%d%d", &a, &b, &c);
        add(a-1, b-1, c);
    }
    solve();
    if (dis2[t] == inf) puts("-1");
    else {
        printf("%d\n", dis2[t]);
        out(n+t);
    }
}
  1. ① 处应填( )
    A. upd(pre[b], n+b, dis[b], q)
    B. upd(a, n+b, d, q)
    C. upd(pre[b], b, dis[b], q)
    D. upd(a, b, d, q)
  2. ② 处应填( )
    A. make_pair(-d, b)
    B. make_pair(d, b)
    C. make_pair(b, d)
    D. make_pair(-b, d)
  3. ③ 处应填( )
    A. 0xff
    B. 0x1f
    C. 0x3f
    D. 0x7f
  4. ④ 处应填( )
    A. upd(a, n+b, dis[a]+c, q)
    B. upd(n+a, n+b, dis2[a]+c, q)
    C. upd(n+a, b, dis2[a]+c, q)
    D. upd(a, b, dis[a]+c, q)
  5. ⑤ 处应填( )
    A. pre2[a%n]
    B. pre[a%n]
    C. pre2[a]
    D. pre[a%n]+1
题解:

本题要求求解有向图中从起点 (s) 到终点 (t) 的严格次短路(长度严格大于最短路)。代码采用 Dijkstra 算法的变种,将每个节点拆分成两个点:原点(对应最短路)和次短点(对应次短路),分别用距离数组 dis[0..n-1]dis[n..2n-1] 记录。通过优先队列同时更新最短路和次短路,并记录前驱以输出路径。

算法思路:
  • 使用双倍点集,分别维护最短路和次短路。
  • 在 Dijkstra 过程中,从队列取出点 (aa),若 (aa < n) 则为原点,否则为次短点。
  • 更新时,若成功更新最短路,则将原最短路转移为次短路候选;若失败,则尝试更新次短路。
  • 最终若次短路存在(dis2[t] != inf),则输出长度并递归输出路径。
填空解析:
  1. :当更新原点 (b) 的最短路时,需要将原有的最短路作为次短路候选,因此调用 upd(pre[b], n+b, dis[b], q),选 A
  2. :优先队列默认大顶堆,为实现小顶堆需存入负距离,因此为 make_pair(-d, b),选 A
  3. :距离数组初始化为极大值,inf = 522133279,(十六进制为 0x1f1f1f1f),,选 B
  4. :当最短路更新失败时,当前距离可能成为次短路,因此尝试用该距离更新次短点,调用 upd(a, n+b, dis[a]+c, q),选 A
  5. :输出次短路路径时,对于次短点 (a)((a \ge n)),其前驱为 pre[a],即 pre2[a%n],选 A

专栏推荐:信奥赛C++提高组csp-s初赛&复赛真题题解(持续更新)
https://blog.csdn.net/weixin_66461496/category_13125089.html


各种学习资料,助力大家一站式学习和提升!!!

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int main(){
	cout<<"##########  一站式掌握信奥赛知识!  ##########";
	cout<<"#############  冲刺信奥赛拿奖!  #############";
	cout<<"######  课程购买后永久学习,不受限制!   ######";
	return 0;
}

1、csp信奥赛高频考点知识详解及案例实践:

CSP信奥赛C++动态规划:
https://blog.csdn.net/weixin_66461496/category_13096895.html点击跳转

CSP信奥赛C++标准模板库STL:
https://blog.csdn.net/weixin_66461496/category_13108077.html 点击跳转

信奥赛C++提高组csp-s知识详解及案例实践:
https://blog.csdn.net/weixin_66461496/category_13113932.html

2、csp信奥赛冲刺一等奖有效刷题题解:

CSP信奥赛C++初赛及复赛高频考点真题解析(持续更新):https://blog.csdn.net/weixin_66461496/category_12808781.html 点击跳转

CSP信奥赛C++一等奖通关刷题题单及题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12673810.html 点击跳转

3、GESP C++考级真题题解:

GESP(C++ 一级+二级+三级)真题题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12858102.html 点击跳转

GESP(C++ 四级+五级+六级)真题题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12869848.html 点击跳转

GESP(C++ 七级+八级)真题题解(持续更新):
https://blog.csdn.net/weixin_66461496/category_13117178.html

4、CSP信奥赛C++竞赛拿奖视频课:

https://edu.csdn.net/course/detail/40437 点击跳转

· 文末祝福 ·

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int main(){
	cout<<"跟着王老师一起学习信奥赛C++";
	cout<<"    成就更好的自己!       ";
	cout<<"  csp信奥赛一等奖属于你!   ";
	return 0;
}
相关推荐
khddvbe19 小时前
C++中的代理模式实战
开发语言·c++·算法
计算机安禾20 小时前
【C语言程序设计】第31篇:指针与函数
c语言·开发语言·数据结构·c++·算法·leetcode·visual studio
酬勤-人间道20 小时前
自研软件模型处理全流程|个人开发经验分享
c++·经验分享·计算机·计算机图形学·桩号·开挖·回填
郝学胜-神的一滴20 小时前
算法奇旅:探寻3/5/7素因子之第k特殊数——优雅的多路指针解法全解析
数据结构·c++·算法·职场和发展
handler0120 小时前
基础算法:分治
c语言·开发语言·c++·笔记·学习·算法·深度优先
2501_9249526920 小时前
设计模式在C++中的实现
开发语言·c++·算法
T1an-120 小时前
(独自升级Lv.2)C++基础面试题
c++
艾莉丝努力练剑20 小时前
【MYSQL】MYSQL学习的一大重点:数据库基础
linux·运维·服务器·数据库·c++·学习·mysql
齐齐大魔王20 小时前
虚拟机网络无法连接
linux·网络·c++·python·ubuntu
2501_9454251521 小时前
C++编译期字符串处理
开发语言·c++·算法