力扣网编程135题:分发糖果(贪心算法)

一. 简介

本文记录力扣网上涉及数组方面的编程题:分发糖果。

这里使用贪心算法的思路来解决(求局部最优,最终求全局最优解):每个孩子只需要考虑与相邻孩子的相对关系。

二. 力扣网编程135题:分发糖果(遍历两遍数组)

n 个孩子站成一排。给你一个整数数组 ratings 表示每个孩子的评分。
你需要按照以下要求,给这些孩子分发糖果:
每个孩子至少分配到 1 个糖果。
相邻两个孩子评分更高的孩子会获得更多的糖果。
请你给每个孩子分发糖果,计算并返回需要准备的 最少糖果数目 。

示例 1:
输入:ratings = [1,0,2]
输出:5
解释:你可以分别给第一个、第二个、第三个孩子分发 2、1、2 颗糖果。

示例 2:
输入:ratings = [1,2,2]
输出:4
解释:你可以分别给第一个、第二个、第三个孩子分发 1、2、1 颗糖果。
第三个孩子只得到 1 颗糖果,这满足题面中的两个条件。

解题思路一:(遍历两遍数组)

我们可以将「相邻的孩子中,评分高的孩子必须获得更多的糖果」这句话拆分为两个规则,分别处理:

左规则:ratings[i-1] < ratings[i],i号孩子的糖果比 i-1号孩子的糖果数量多;

右规则:ratings[i] > ratings[i+1],i号孩子比 i+1号孩子的糖果数量少;

遍历该数组两次,处理出每一个学生分别满足左规则或右规则时,最少需要被分得的糖果数量。每个人最终分得的糖果数量即为这两个数量的最大值。

具体方法如下:
  1. 定义一个数组left[ratingsSize],用于存放满足左规则的糖果数目;变量 right 存放满足右规则的糖果数目;

  2. 满足左规则:从前往后遍历数组,若 ratings[i-1] < ratings[i],则 left[i] = left[i-1] +1,否则,left[i] = 1;

  3. 满足右规则:类似处理,不过是从后往前进行遍历。

若 ratings[i] > ratings[i+1],则 right += 1,否则,right =1。在遍历过程中,求 满足左规则和右规则的糖果数目进行比较,求最大值,同时进行累计。

C语言实现如下:

复制代码
//遍历两次数组,满足条件
//1.左规则:ratings[i-1] < ratings[i], 则ratings[i]=ratings[i-1]+1
//2.右规则:ratings[i] > ratings[i+1],则ratings[i]= ratings[i+1]+1
int candy(int* ratings, int ratingsSize) {
    int i;
    int count = 0;
    int left[ratingsSize];

    //从前往后遍历
    //左规则:ratings[i-1] < ratings[i], 则ratings[i]=ratings[i-1]+1
    for(i = 0; i < ratingsSize; i++) {
        if((i>0) && (ratings[i-1] < ratings[i])) {
            left[i] = left[i-1]+1;
        }
        else {
            left[i] = 1;
        }
    }
    int right = 0;
    //从后往前遍历
    //右规则:ratings[i] > ratings[i+1],则ratings[i]= ratings[i+1]+1
    for(i = ratingsSize-1; i >= 0; i--) {
        if((i < ratingsSize-1) && (ratings[i] > ratings[i+1])) {
            right += 1;
        } 
        else {
            right = 1;
        }
        count += fmax(left[i], right);
    } 
    return count;
}

可以看出,贪心算法的解法的时间复杂度为 O(n),空间负责度为O(n)。

第二种实现如下,也是同样的思路(贪心算法):

复制代码
//贪心算法:
//局部最优:每个孩子只需要关心和她相邻的关系
int candy(int* ratings, int ratingsSize) {
    int i;
    int candies[ratingsSize];
    int right = 0;
    int count = 0;

    memset(candies, 0, ratingsSize*sizeof(int));

    for(i = 0; i < ratingsSize; i++) {
        candies[i] = 1;
    }

    //遵循左规则:从左到右遍历
    //如果 ratings[i-1] < ratings[i]
    for(i = 0; i < ratingsSize; i++) {
        if((i>0) && (ratings[i-1] < ratings[i])) {
            candies[i] = candies[i-1] + 1;
        }
    }

    //遵循右规则:从右向左遍历
    //如果 ratings[i] > ratings[i+1]
    for(i = ratingsSize-1; i>=0; i--) {
        if((i<ratingsSize-1) && (ratings[i] > ratings[i+1])) {
            right += 1;
        }
        else {
            right = 1;
        }
        count += fmax(candies[i], right);
    }

    return count;
}
相关推荐
AMiner:AI科研助手4 分钟前
警惕!你和ChatGPT的对话,可能正在制造分布式妄想
人工智能·分布式·算法·chatgpt·deepseek
CHANG_THE_WORLD3 小时前
并发编程指南 同步操作与强制排序
开发语言·c++·算法
gaoshou454 小时前
代码随想录训练营第三十一天|LeetCode56.合并区间、LeetCode738.单调递增的数字
数据结构·算法
自信的小螺丝钉5 小时前
Leetcode 240. 搜索二维矩阵 II 矩阵 / 二分
算法·leetcode·矩阵
KING BOB!!!6 小时前
Leetcode高频 SQL 50 题(基础版)题目记录
sql·mysql·算法·leetcode
我是渣哥6 小时前
Java String vs StringBuilder vs StringBuffer:一个性能优化的探险故事
java·开发语言·jvm·后端·算法·职场和发展·性能优化
THMAIL6 小时前
机器学习从入门到精通 - 机器学习调参终极手册:网格搜索、贝叶斯优化实战
人工智能·python·算法·机器学习·支持向量机·数据挖掘·逻辑回归
lytk997 小时前
矩阵中寻找好子矩阵
线性代数·算法·矩阵
珊瑚怪人7 小时前
算法随笔(一)
算法
晚安里7 小时前
JVM相关 4|JVM调优与常见参数(如 -Xms、-Xmx、-XX:+PrintGCDetails) 的必会知识点汇总
java·开发语言·jvm·后端·算法