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

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

3个月前 (07-16)

洛谷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避免大数溢出。此思路适用于类似“相邻约束优化”的问题,为算法竞赛中常见的解题模式。


原创内容 转载请注明出处

分享给朋友:

相关文章

IOI 1994 洛谷1216:如何用动态规划高效解决数字三角形问题?附完整代码解析

IOI 1994 洛谷1216:如何用动态规划高效解决数字三角形问题?附完整代码解析

题目重解给定一个由数字组成的三角形结构,从顶部出发,每次可以移动到下方相邻的数字,最终到达底部。我们的目标是找到一条路径,使得路径上经过的数字总和最大。这个问题在实际中有许多应用场景,如最优路径规划、...

力扣144:递归之美 轻松掌握二叉树前序遍历

力扣144:递归之美 轻松掌握二叉树前序遍历

题目解读二叉树的前序遍历是一种基础但重要的树遍历方式,其遍历顺序为:先访问根节点,然后递归地前序遍历左子树,最后递归地前序遍历右子树。给定一个二叉树的根节点,我们需要按照这个顺序访问所有节点,并将它们...

力扣540题:线性扫描法如何高效定位唯一数

力扣540题:线性扫描法如何高效定位唯一数

题目重解一个严格递增的有序数组中,除某个元素外,其余每个元素均出现两次。这个看似简单的条件背后隐藏着巧妙的规律——单一元素会打破数组的"成对对称性"。题目要求以O(log n)时间...

手搓顺序表类代码注释与详解:从零实现动态数组(新手教程)

一、简介和特点顺序表(Sequential List)是数据结构中基础的一种线性表,其特点是将数据元素存储在连续的内存空间中。通过数组实现,支持随机访问(即通过索引直接访问元素),适用于频繁随机读取的...

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

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

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

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

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

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

发表评论

访客

看不清,换一张

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