当前位置:首页 > 入门组 > CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

6个月前 (06-15)

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解  公交换乘题 队列优化 动态规划 第1张

一、题目解读

CSP-J 2019年的“公交换乘”题目(洛谷P5661)要求模拟地铁与公交交替出行的费用计算。题目核心在于地铁消费会产生优惠券,而公交可在45分钟内使用优惠券抵扣车费。需要处理n条出行记录,优化总费用。该问题考察对时间窗口与动态资源管理的理解,需平衡实时状态更新与历史数据利用。

二、解题思路

代码采用“队列+动态规划”策略:

    1. 使用队列存储优惠券,遵循“先进先出”原则,确保优惠券按获取时间排序

    2. 地铁消费时生成新券并入队,公交消费时检查队首券是否过期:

        过期券出队,避免无效抵扣。

        遍历剩余券,优先使用面值≥公交费用的券,首次匹配成功即停止(贪心策略)。

    3. 未使用券暂存临时队列,最终恢复至主队列,维持原顺序。

此思路将时间复杂度控制在O(n),避免重复遍历,同时保证状态一致性。

三、解题步骤

1. 输入处理:读取n条记录,每条含类型(0/1)、费用、时间。

2. 地铁处理(type=0):

    累加费用至总账。

    生成新券(面值=费用,时间=当前时间)入队。

3. 公交处理(type=1):

    清理过期券。

    循环优惠券队列:

        若券面值≥公交费且未使用过,标记使用并跳出循环。

        否则将券暂存临时队列。

        若未找到可用券,累加公交费至总账。

    将临时队列券恢复至主队列。

4. 输出总费用。

四、代码与注释

#include <iostream>  
#include <queue>  
using namespace std;  

struct Coupon {  
    int price;  // 地铁票价(优惠券面值)  
    int time;   // 获得优惠券的时间  
};  

int main() {  
    ios::sync_with_stdio(false);  // 加快输入输出  
    cin.tie(nullptr);           
  
    int n, total = 0;  
    cin >> n;  
    queue<Coupon> coupons;  // 优惠券队列(先进先出)  

    for (int i = 0; i < n; ++i) {  
        int type, price, time;  
        cin >> type >> price >> time;  

        if (type == 0) {  // 地铁记录  
            total += price;  // 地铁必须付费  
            coupons.push({price, time});  // 生成优惠券  
        }  
        else {  // 公交记录  
            // 移除过期优惠券(队首是最早的)  
            while (!coupons.empty() && time - coupons.front().time > 45) {  
                coupons.pop();  
            }  

            bool used = false;  
            // 临时队列用于恢复未使用的优惠券  
            queue<Coupon> temp;  

            // 尝试使用优惠券  
            while (!coupons.empty()) {  
                Coupon c = coupons.front();  
                coupons.pop();  

                if (!used && c.price >= price) {  // 找到可用优惠券  
                    used = true;  // 标记已使用  
                }  
                else {  // 未使用的优惠券暂存  
                    temp.push(c);  
                }  
            }  

            // 恢复未使用的优惠券  
            while (!temp.empty()) {  
                coupons.push(temp.front());  
                temp.pop();  
            }  

            if (!used) total += price;  // 没有可用优惠券则付费  
        }  
    }  

    cout << total << endl;  
    return 0;  
}

代码核心逻辑:通过队列维护优惠券时效性,利用贪心策略优先消耗高面值券,确保每个公交费用最小化。

五、总结

该解法巧妙运用队列实现“时间窗口”管理,结合动态规划思想降低复杂度。关键点在于:

    1. 优惠券队列的FIFO特性自动维护时效。

    2. 临时队列保障未使用券的状态还原。

    3. 单次公交消费仅需遍历当前有效券,避免全局搜索。

此思路为处理带时间限制的资源复用问题提供了经典模板,适用于类似场景的算法设计。

原创内容 转载请注明出处

分享给朋友:

相关文章

从零到一掌握背包问题:洛谷P1164题解精讲,附带优化

从零到一掌握背包问题:洛谷P1164题解精讲,附带优化

题目重解:小A带着m元钱来到餐馆,菜单上有n道菜,每道菜都有确定的价格。现在需要计算出刚好花完m元的点菜方案总数。这个问题看似简单,但当菜品数量增多时,暴力枚举就会变得不可行,需要更高效的算法来解决。...

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

一、题目解读洛谷P4551题要求在一个无向图中,寻找任意两点路径权值异或后的最大值。题目输入为图的边信息(点数n和n-1条边),每条边包含起点、终点及权值。需输出所有路径中权值异或的最大值。问题核心在...

洛谷2789题解:直线交点数的递归求解与优化(附代码详解)

洛谷2789题解:直线交点数的递归求解与优化(附代码详解)

一、题目解读洛谷2789题要求计算n条直线在平面上两两相交时产生的不同交点数量。题目强调“不同”交点,需排除重复情况。解题关键在于如何高效枚举所有可能的交点组合,并避免重复计数。二、解题思路参考代码采...

牛客网288555题解题指南:动态规划求解小红的暑假(附代码解析)

牛客网288555题解题指南:动态规划求解小红的暑假(附代码解析)

一、题目解读牛客网288555题要求解决一个组合数学问题:有三位朋友,每天需邀请其中一位参加聚会,但不能连续两天邀请同一位朋友。给定天数n,求满足条件的不同邀请方案总数。题目考察动态规划、状态转移及组...

【2020蓝桥杯国赛C组】补给题解析:从Floyd到动态规划的高效解法

【2020蓝桥杯国赛C组】补给题解析:从Floyd到动态规划的高效解法

一、题目解读2020年蓝桥杯国赛C组“补给”题目要求:给定n个村庄坐标及最大补给距离D,需判断是否所有村庄均可从总部(村庄0)直接或间接到达,并计算访问所有村庄的最小路径(即旅行商问题TSP)。题目核...

牛客226516题解:动态规划解决完全背包问题(附代码解析)

牛客226516题解:动态规划解决完全背包问题(附代码解析)

一、题目解读牛客226516题要求解决两类完全背包问题:普通完全背包(求最大价值)与恰好装满的完全背包(判断是否存在方案)。题目给定n个物品,每个物品有体积v和价值w,背包容量为V。需分别计算两种情况...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。