LeetCode159--完全平方数(L279)、寻找重复数(L287)

本文探讨了LeetCode中两道题目——寻找构成n的最少完全平方数数量及找出数组中重复的唯一整数。对于完全平方数问题,采用广度优先搜索或动态规划的方法,缩小范围在[1, 根号n]内。寻找重复数的解决方案利用位运算和双指针结合二分查找,找到满足cnt[i]条件的重复数。" 88218219,7474554,Python初学者实战项目指南,"['Python学习', 'Python爬虫', 'web开发']
摘要由CSDN通过智能技术生成

1、完全平方数

//给定正整数 n,找到若干个完全平方数(比如 1, 4, 9, 16, …)使得它们的和等于 n。你需要让组成和的完全平方数的个数最少。
//
// 给你一个整数 n ,返回和为 n 的完全平方数的 最少数量 。
//
// 完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,1、4、9 和 16 都是完全平方数,而 3 和 11 不是。
//
//
//
//
// 示例 1:
//
//
//输入:n = 12
//输出:3
//解释:12 = 4 + 4 + 4
//
// 示例 2:
//
//
//输入:n = 13
//输出:2
//解释:13 = 4 + 9
//
//
// 提示:
//
//
// 1 <= n <= 104
//
// Related Topics 广度优先搜索 数学 动态规划

我们维护一个数组,代表i最少能以dp[i]个完全平方数代替。我们通过不断枚举来缩小范围,,因为输入的数是n,所以完全平方数一定是在[1,根号n]的范围内,比如现在我们枚举到了j,剩下的数就是n-j的平方,然后我们现在的目标就是寻找最小数量的完全平方数来对实现n-j的平方,这样问题的范围就被缩小了。最后得到的转移方程为:
在这里插入图片描述

public int numSquares(int n) {
        //求出每个数最少可以由几个完全平方数替代,然后往后推影响后面的数。
        int[] f = new int[n+1];
        for (int i = 1; i <= n; i++) {
            int minn = Integer.MAX_VALUE;
            for (int j = 1; j * j <= i; j++) {
                minn = Math.min(minn, f[i-j*j]);
            }
            f[i] = minn + 1;
        }
        return f[n];
    }

2、寻找重复数

//给定一个包含 n + 1 个整数的数组 nums ,其数字都在 1 到 n 之间(包括 1 和 n),可知至少存在一个重复的整数。
//
// 假设 nums 只有 一个重复的整数 ,找出 这个重复的数 。
//
// 你设计的解决方案必须不修改数组 nums 且只用常量级 O(1) 的额外空间。
//
//
//
// 示例 1:
//
//
//输入:nums = [1,3,4,2,2]
//输出:2
//
//
// 示例 2:
//
//
//输入:nums = [3,1,3,4,2]
//输出:3
//
//
// 示例 3:
//
//
//输入:nums = [1,1]
//输出:1
//
//
// 示例 4:
//
//
//输入:nums = [1,1,2]
//输出:1
//
//
//
//
// 提示:
//
//
// 1 <= n <= 105
// nums.length == n + 1
// 1 <= nums[i] <= n
// nums 中 只有一个整数 出现 两次或多次 ,其余整数均只出现 一次
//
//
//
//
// 进阶:
//
//
// 如何证明 nums 中至少存在一个重复的数字?
// 你可以设计一个线性级时间复杂度 O(n) 的解决方案吗?
//
// Related Topics 位运算 数组 双指针 二分查找

先找规律,一共只有n+1个数,数字都在[1,n]之间,而且只有一个数被重复两次或者多次,那么就只有两种情况:

  1. 即其中有一个数出现了两次;
  2. 其中有一个数字出现了多次(超过两次);
i123456
nums[i]123345

假设我们重复的数是target,然后cnt[i]表示nums数组中小于等于i的数有多少个。那么在[1,target-1]这个范围内满足cnt[i]<=i,而在[target,n]这个范围内满足cnt[i]>i,如下:

i12345
cnt[i]12456

那么再思考如果一个数出现三次及以上还符不符合这个规则:

i1234567
nums[i]1233346

可见一个数出现三次及以上肯定是顶替了其他的数,比如上面3就多出现了一次来替换了5。我们假设在被替换的数为j,那么在target和j-1之间的所有cnt值都会加一,依然满足cnt[i]>i这个条件,如果被替换的数小于target也可以思考一下,就相当于被替换的数到target-1这个范围内的cnt值都见了一,依然符合cnt[i]<=i这个条件。所以我们利用这个条件进行二分查找,找到两个条件的交界处。

class Solution {
    public int findDuplicate(int[] nums) {
        int n = nums.length;
        int l = 1, r = n - 1, ans = -1;
        while (l <= r) {
            int mid = (l + r) >> 1;
            int cnt = 0;
            for (int i = 0; i < n; ++i) {
                if (nums[i] <= mid) {
                    cnt++;
                }
            }
            if (cnt <= mid) {
                l = mid + 1;
            } else {
                r = mid - 1;
                ans = mid;
            }
        }
        return ans;
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值