题目: 假设你是一位很棒的家长,想要给你的孩子们一些小饼干。但是,每个孩子最多只能给一块饼干。 对每个孩子 i,都有一个胃口值 g[i],这是能让孩子们满足胃口的饼干的最小尺寸;并且每块饼干 j, 都有一个尺寸 s[j] 。如果 s[j] >= g[i],我们可以将这个饼干 j 分配给孩子 i ,这个孩子会得到满足。 你的目标是尽可能满足越多数量的孩子,并输出这个最大数值。 提示: 1 <= g.length <= 3 * 10^4 0 <= s.length <= 3 * 10^4 1 <= g[i], s[j] <= 2^31 - 1 --------------- 示例: 输入: g = [1,2,3], s = [1,1] 输出: 1 解释: 你有三个孩子和两块小饼干,3个孩子的胃口值分别是:1,2,3。 虽然你有两块小饼干,由于他们的尺寸都是1,你只能让胃口值是1的孩子满足。 所以你应该输出1。 输入: g = [1,2], s = [1,2,3] 输出: 2 解释: 你有两个孩子和三块小饼干,2个孩子的胃口值分别是1,2。 你拥有的饼干数量和尺寸都足以让所有孩子满足。 所以你应该输出2. --------------- 思考: 为了了满足更多的小孩,就不要造成饼干尺寸的浪费。 大尺寸的饼干既可以满足胃口大的孩子也可以满足胃口小的孩子,那么就应该优先满足胃口大的。 这里的局部最优就是大饼干喂给胃口大的,充分利用饼干尺寸喂饱一个,全局最优就是喂饱尽可能多的小孩!! 可以尝试使用贪心策略,先将饼干数组和小孩数组排序。 然后从后向前遍历小孩数组,用大饼干优先满足胃口大的,并统计满足小孩数量。 如图:
这个例子可以看出饼干9只有喂给胃口为7的小孩,这样才是整体最优解,并想不出反例,那么就可以撸代码了。 从代码中可以看出我用了一个index来控制饼干数组的遍历,遍历饼干并没有再起一个for循环,而是采用自减的方式,这也是常用的技巧。 有的同学看到要遍历两个数组,就想到用两个for循环,那样逻辑其实就复杂了。 也可以换一个思路,小饼干先喂饱小胃口 ----------------------- 代码: 解法一: // 思路1:优先考虑饼干,小饼干先喂饱小胃口 class Solution{ public int findContentChildren(int[] g,int[] s){ //先排序 Arrays.sort(g); Arrays.sort(s); //定义胃口数组g 的 指针 int start = 0; //计数 int count = 0; for(int i = 0;i=g[start]){ start++; count++; } } return count; } } 解法二: // 思路2:优先考虑胃口,先喂饱大胃口 class Solution{ public int findContentChildren(int[] g,int[] s){ //先排序 Arrays.sort(g); Arrays.sort(s); //定义饼干数组的指针 int start = s.length-1; //计数 int count = 0; for(int index = g.length-1;index>=0;index--){ if(start>=0 && s[start]>=g[index]){ start--; count++; } } return count; } }
力扣
贪心概念: 1、什么是贪心: 贪心的本质是选择每一阶段的局部最优,从而达到全局最优。 这么说有点抽象,来举一个例子: 例如,有一堆钞票,你可以拿走十张,如果想达到最大的金额,你要怎么拿? 指定每次拿最大的,最终结果就是拿走最大数额的钱。 每次拿最大的就是局部最优,最后拿走最大数额的钱就是推出全局最优。 再举一个例子如果是 有一堆盒子,你有一个背包体积为n,如何把背包尽可能装满, 如果还每次选最大的盒子,就不行了。这时候就需要动态规划。 2、什么时候用贪心: 很多同学做贪心的题目的时候,想不出来是贪心,想知道有没有什么套路可以一看就看出来是贪心。 说实话贪心算法并没有固定的套路。 所以唯一的难点就是如何通过局部最优,推出整体最优。 那么如何能看出局部最优是否能推出整体最优呢?有没有什么固定策略或者套路呢? 不好意思,也没有! 靠自己手动模拟,如果模拟可行,就可以试一试贪心策略,如果不可行,可能需要动态规划。 有同学问了如何验证可不可以用贪心算法呢? 最好用的策略就是举反例,如果想不到反例,那么就试一试贪心吧。 可有有同学认为手动模拟,举例子得出的结论不靠谱,想要严格的数学证明。 一般数学证明有如下两种方法: 数学归纳法 反证法 看教课书上讲解贪心可以是一堆公式,估计大家连看都不想看,所以数学证明就不在我要讲解的范围内了, 大家感兴趣可以自行查找资料。 面试中基本不会让面试者现场证明贪心的合理性,代码写出来跑过测试用例即可,或者自己能自圆其说理由就行了。 举一个不太恰当的例子:我要用一下1+1 = 2,但我要先证明1+1 为什么等于2。严谨是严谨了,但没必要。 虽然这个例子很极端,但可以表达这么个意思:刷题或者面试的时候, 手动模拟一下感觉可以局部最优推出整体最优,而且想不到反例,那么就试一试贪心。 例如刚刚举的拿钞票的例子,就是模拟一下每次拿做大的, 最后就能拿到最多的钱,这还要数学证明的话,其实就不在算法面试的范围内了,可以看看专业的数学书籍! 所以这也是为什么很多同学通过(accept)了贪心的题目,但都不知道自己用了贪心算法, 因为贪心有时候就是常识性的推导,所以会认为本应该就这么做! 那么刷题的时候什么时候真的需要数学推导呢? 例如这道题目:链表:环找到了,那入口呢? 这道题不用数学推导一下,就找不出环的起始位置, 想试一下就不知道怎么试,这种题目确实需要数学简单推导一下。 3、贪心一般解题步骤: 贪心算法一般分为如下四步: 将问题分解为若干个子问题 找出适合的贪心策略 求解每一个子问题的最优解 将局部最优解堆叠成全局最优解 其实这个分的有点细了,真正做题的时候很难分出这么详细的解题步骤, 可能就是因为贪心的题目往往还和其他方面的知识混在一起
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)