LeetCode 453. 最小操作次数使数组元素相等 / 66. 加一 / 229. 求众数 II(摩尔投票法)

453. 最小操作次数使数组元素相等

2021.10.20 每日一题

题目描述

给你一个长度为 n 的整数数组,每次操作将会使 n - 1 个元素增加 1 。返回让数组所有元素相等的最小操作次数。

示例 1:

输入:nums = [1,2,3]
输出:3
解释:
只需要3次操作(注意每次操作会增加两个元素的值):
[1,2,3] => [2,3,3] => [3,4,3] => [4,4,4]

示例 2:

输入:nums = [1,1,1]
输出:0

提示:

n == nums.length
1 <= nums.length <= 10^5
-10^9 <= nums[i] <= 10^9
答案保证符合 32-bit 整数

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/minimum-moves-to-equal-array-elements
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

每次都会有n-1个数字加1,所以考虑这组数字的最小值min,先让最小值加到最大值max,然后此时次大值也加了max - min,这时,次大值变成了最大值,然后再让最小值增加到最大值,以此类推,可以发现规律
或者说让其他数加一,相当于让一个数减1,那么怎么才能让所有数相同呢,就是让所有数都减到和最小数相同

class Solution {
    public int minMoves(int[] nums) {
        //这样想了一下,首先,最小值要达到最大值要max-min次
        //然后,第二大值肯定会超过最大值,那么,要达到第二大值,又要,第二大-min次
        //然后,又有超过的
        //比如, 1 2 5 8,先加7次,得到 8 9 12 8,然后,加4次,得到12 13 12 12
        //然后加一次,得到,13 13 13 13
        //所以最后结果就是 2 + 5 + 8 - 3 = 12
        int l = nums.length;
        int min = Integer.MAX_VALUE;
        int idx = 0;
        for(int i = 0; i < l; i++){
            if(nums[i] < min){
                idx = i;
                min = nums[i];
            }
        }
        int res = 0;
        for(int i = 0; i < l; i++){
            res += nums[i] - min;
        }
        return res;
    }
}

用流的写法

class Solution {
    public int minMoves(int[] nums) {
        int minVal = Arrays.stream(nums).min().getAsInt();
        int sum = Arrays.stream(nums).sum();
        return sum - minVal * nums.length;
    }
}

66. 加一

2021.10.21 每日一题

题目描述

给定一个由 整数 组成的 非空 数组所表示的非负整数,在该数的基础上加一。

最高位数字存放在数组的首位, 数组中每个元素只存储单个数字。

你可以假设除了整数 0 之外,这个整数不会以零开头。

示例 1:

输入:digits = [1,2,3]
输出:[1,2,4]
解释:输入数组表示数字 123。

示例 2:

输入:digits = [4,3,2,1]
输出:[4,3,2,2]
解释:输入数组表示数字 4321。

示例 3:

输入:digits = [0]
输出:[1]

提示:

1 <= digits.length <= 100
0 <= digits[i] <= 9

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/plus-one
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

模拟就行了
如果是9那就进位,如果不是,直接当前为加1然后输出
如果都是9,那么就新开一个数组,然后第一位为1

class Solution {
    public int[] plusOne(int[] digits) {
        int l = digits.length;
        for(int i = l - 1; i >= 0; i--){
            if(digits[i] == 9){
                digits[i] = 0;
            }else{
                digits[i] = digits[i] + 1;
                return digits;
            }
        }
        int[] res = new int[l + 1];
        res[0] = 1;
    
        return res;
    }
}

229. 求众数 II

2021.10.22 每日一题

题目描述

给定一个大小为 n 的整数数组,找出其中所有出现超过 ⌊ n/3 ⌋ 次的元素。

示例 1:

输入:[3,2,3]
输出:[3]

示例 2:

输入:nums = [1]
输出:[1]

示例 3:

输入:[1,1,1,3,3,2,2,2]
输出:[1,2]

提示:

1 <= nums.length <= 5 * 10^4
-10^9 <= nums[i] <= 10^9

进阶:尝试设计时间复杂度为 O(n)、空间复杂度为 O(1)的算法解决此问题。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/majority-element-ii
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

摩尔投票法的进阶,两个候选人的摩尔投票法
如果给候选人的其中一个投票,那么候选人票数加1
如果给别人投票,那么两个候选人的票数都减1
如果票数为0的,那么更换候选人
最后,要判断选出来的候选人是否满足大于1/3票数的要求,因为题目并没有保证有

class Solution {
    public List<Integer> majorityElement(int[] nums) {
        //摩尔投票法(升级),我理解这个方法就是,选n个候选人,就需要有多于 1 / (n + 1)的票数,
        //所以将剩下的小于1/n的人和这几个候选人每个人的票数相加,就相当于每个候选人的票数大于这一部分的1/2,
        //所以就可以转换为169题,选大于1/2的那个题
        //所以先确定两个候选人,如果第三个投票结果和他们中任何一个相同,使其中那个人的票数加1;
        //如果第三个投票结果和他们任何一个都不同,那么就都抵消掉
        //以这个过程遍历结束,剩余结果可能是符合要求的,也可能是不符合的
        //所以需要计数阶段,就是遍历一次nums,统计个数

        List<Integer> res = new ArrayList<>();
        int cand1 = nums[0], count1 = 0;
        int cand2 = nums[0], count2 = 0;
        //抵消阶段
        for(int n : nums){
            if(n == cand1){
                count1++;
                continue;
            }
            if(n == cand2){
                count2++;
                continue;
            }
            //当count1等于0的时候,换候选人
            if(count1 == 0){
                cand1 = n;
                count1++;
                continue;
            }
            if(count2 == 0){
                cand2 = n;
                count2++;
                continue;
            }
            //如果不相同,就减去
            count1--;
            count2--;
        }
        //计数阶段
        count1 = 0;
        count2 = 0;
        for(int n : nums){
            if(cand1 == n)
                count1++;
            //这里必须用else if,因为两个候选人可能相同
            else if(cand2 == n){
                count2++;
            }
        }
        if(count1 > nums.length / 3)
            res.add(cand1);
        if(count2 > nums.length / 3)
            res.add(cand2);
        return res;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 好的,我来用中文回复这个链接:https://leetcode-cn.com/tag/dynamic-programming/ 这个链接是 LeetCode 上关于动态规划的题目集合。动态规划是一种常用的算思想,可以用来解决很多实际问题,比如最长公共子序列、背包问题、最短路径等等。在 LeetCode 上,动态规划也是一个非常重要的题型,很多题目都需要用到动态规划的思想来解决。 这个链接里包含了很多关于动态规划的题目,按照难度从简单到困难排列。每个题目都有详细的题目描述、输入输出样例、题目解析和代码实现等内容,非常适合想要学习动态规划算的人来练习和提高自己的能力。 总之,这个链接是一个非常好的学习动态规划算的资源,建议大家多多利用。 ### 回答2: 动态规划是一种算思想,通常用于优化具有重叠子问题和最优子结构性质的问题。由于其成熟的数学理论和强大的实用效果,动态规划在计算机科学、数学、经济学、管理学等领域均有重要应用。 在计算机科学领域,动态规划常用于解决最优化问题,如背包问题、图像处理、语音识别、自然语言处理等。同时,在计算机网络和分布式系统中,动态规划也广泛应用于各种优化算中,如链路优化、路由算、网络流量控制等。 对于算领域的程序员而言,动态规划是一种必要的技能和知识点。在LeetCode这样的程序员平台上,题目分类和标签设置十分细致和方便,方便程序员查找并深入学习不同类型的算LeetCode的动态规划标签下的题目涵盖了各种难度级别和场景的问题。从简单的斐波那契数列、迷宫问题到可以用于实际应用的背包问题、最长公共子序列等,难度不断递进且话题丰富,有助于开发人员掌握动态规划的实际应用技能和抽象思维模式。 因此,深入LeetCode动态规划分类下的题目学习和练习,对于程序员的职业发展和技能提升有着重要的意义。 ### 回答3: 动态规划是一种常见的算思想,它通过将问题拆分成子问题的方式进行解。在LeetCode中,动态规划标签涵盖了众多经典和优美的算问题,例如斐波那契数列、矩阵链乘、背包问题等。 动态规划的核心思想是“记忆化搜索”,即将中间状态保存下来,避免重复计算。通常情况下,我们会使用一张二维表来记录状态转移过程中的中间值,例如动态规划解斐波那契数列问题时,就可以定义一个二维数组f[i][j],代表第i项斐波那契数列中,第j个元素的值。 在LeetCode中,动态规划标签下有众多难度不同的问题。例如,经典的“爬楼梯”问题,要我们计算到n级楼梯的方案数。这个问题的解非常简单,只需要维护一个长度为n的数组,记录到达每一级楼梯的方案数即可。类似的问题还有“零钱兑换”、“乘积最大子数组”、“通配符匹配”等,它们都采用了类似的动态规划思想,通过拆分问题、保存中间状态来解问题。 需要注意的是,动态规划算并不是万能的,它虽然可以处理众多经典问题,但在某些场景下并不适用。例如,某些问题的状态转移过程比较复杂,或者状态转移方程中存在多个参数,这些情况下使用动态规划算可能会变得比较麻烦。此外,动态规划算也存在一些常见误区,例如错用贪心思想、未考虑边界情况等。 总之,掌握动态规划算对于LeetCode的学习和解题都非常重要。除了刷题以外,我们还可以通过阅读经典的动态规划书籍,例如《算竞赛进阶指南》、《算与数据结构基础》等,来深入理解这种算思想。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值