当前位置:首页 > 洛谷 > 洛谷P3817题解:基于贪心算法的糖果分配优化策略

洛谷P3817题解:基于贪心算法的糖果分配优化策略

1天前

洛谷P3817题解:基于贪心算法的糖果分配优化策略 洛谷题解 贪心算法 C++ 第1张

一、题目解读

洛谷P3817题目要求处理n个盒子中的糖果分配问题:给定每个盒子的糖果数a[i]及限制值x,需通过吃掉部分糖果,确保任意相邻盒子(a[i-1] + a[i])的总和不超过x。输出最小需吃掉的糖果总数。题目关键在于高效处理相邻关系,避免全局遍历,降低时间复杂度。

二、解题思路

本题采用贪心策略的核心逻辑:局部最优解可推导全局最优。由于需最小化总消耗,每次处理相邻盒子时,优先减少当前盒子(即a[i])的糖果数,因其能直接影响下一对相邻盒子的判断。若当前减少量不足,再补充减少前一个盒子(即a[i-1])的糖果。该策略确保每一步的“损耗”对后续影响最大化,从而避免回溯或复杂计算。

三、解题步骤

1. 输入与初始化:读取n、x及盒子数组a[],初始化总消耗total_eaten=0。

2. 遍历处理:从左到右遍历相邻盒子对(i-1,i),若a[i-1] + a[i] > x,则需调整:

○ 计算需减少的总量:need_to_eat = (a[i-1] + a[i]) - x。

○ 优先从a[i]中减少,取min(need_to_eat, a[i])以避免过量。

○ 若仍需调整,从a[i-1]中补充减少量,更新total_eaten。

3. 输出结果:最终total_eaten即为最小消耗。

四、代码与注释

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int main() {
    int n, x;
    cin >> n >> x; // 输入盒子数量n和限制值x
    
    vector<int> a(n); // 存储每个盒子的糖果数
    for (int i = 0; i < n; ++i) {
        cin >> a[i];
    }
    
    long long total_eaten = 0; // 记录总消耗(需long long防溢出)
    
    // 从左到右处理相邻盒子
    for (int i = 1; i < n; ++i) {
        if (a[i-1] + a[i] > x) { // 若相邻和超限
            int need_to_eat = a[i-1] + a[i] - x; // 计算需减少的总量
            int can_eat = min(need_to_eat, a[i]); // 优先从当前盒子减少
            a[i] -= can_eat; // 更新当前盒子糖果数
            if (can_eat < need_to_eat) { // 若仍需调整
                a[i-1] -= (need_to_eat - can_eat); // 从前一个盒子补充减少
            }
            total_eaten += need_to_eat; // 累加总消耗
        }
    }
    
    cout << total_eaten << endl; // 输出结果
    return 0;
}

五、总结

该解法通过贪心算法实现线性时间复杂度O(n),核心在于“优先当前盒子减少”的策略,确保局部决策的最优性。代码简洁且易于理解,无需复杂数据结构动态规划。注意变量类型需使用long long避免大数溢出。此思路适用于类似“相邻约束优化”的问题,为算法竞赛中常见的解题模式。


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣740.删除并获得点数 预处理与动态规划的巧妙融合

力扣740.删除并获得点数 预处理与动态规划的巧妙融合

题意解析:给定一组数字,每当你选择一个数字x时,所有等于x-1和x+1的数字都会被自动移除。你需要通过巧妙的选择顺序,最大化获得的点数总和。这个问题可以转化为对离散化数字分布的动态规划问题——将相邻数...

力扣第92题:三步定位 精准反转链表指定区间

力扣第92题:三步定位 精准反转链表指定区间

题目解读给定一个单链表和两个整数left与right,要求将链表中从第left个节点到第right个节点的部分进行反转,而保持其他部分不变。例如,对于链表1→2→3→4→5,left=2,right=...

牛客14496题解:括号最大深度问题(栈思想与代码优化)

牛客14496题解:括号最大深度问题(栈思想与代码优化)

一、题目解读牛客14496题要求计算给定括号字符串中的最大深度。例如,对于字符串 "(()())",最大深度为2。题目考察对括号嵌套结构的理解,以及如何通过编程找到最深嵌套层次。二...

力扣3112题解法:带时间限制的最短路径问题解析(C++代码)

力扣3112题解法:带时间限制的最短路径问题解析(C++代码)

一、题目解读力扣3112题要求解决带时间限制的最短路径问题:给定一个有向图,节点具有消失时间,需计算从起点到各节点的最短路径,且路径总时间不能超过节点的消失时间。题目难点在于需在传统最短路径算法(如D...

【洛谷1184题解析】用C++高效解决地点匹配问题(附代码与解题思路)

【洛谷1184题解析】用C++高效解决地点匹配问题(附代码与解题思路)

一、题目解读洛谷1184题要求处理一组地点列表与行程记录,统计其中匹配的天数。题目难点在于高效处理带有空格的字符串输入,以及快速判断每日行程是否在高手可去地点集合中。需要兼顾输入格式解析与算法效率。二...

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

一、题目解读洛谷1220题要求计算在n个位置放置灯的情况下,通过关闭连续区间灯并移动至区间端点,使得总耗电量最小。需考虑灯的功率与位置差异,设计高效的算法求解最优策略。二、解题思路1. 动态规划 +...

发表评论

访客

看不清,换一张

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