给你一个正整数数组 nums
,你需要从中任选一些子集,然后将子集中每一个数乘以一个 任意整数,并求出他们的和。
假如该和结果为 1
,那么原数组就是一个「好数组」,则返回 True
;否则请返回 False
。
示例 1:
输入:nums = [12,5,7,23] 输出:true 解释:挑选数字 5 和 7。 5*3 + 7*(-2) = 1
示例 2:
输入:nums = [29,6,10] 输出:true 解释:挑选数字 29, 6 和 10。 29*1 + 6*(-3) + 10*(-1) = 1
示例 3:
输入:nums = [3,6] 输出:false
提示:
1 <= nums.length <= 10^5
1 <= nums[i] <= 10^9
方法一:数学(裴蜀定理)
我们先可以考虑选取两个数的情况,若选取的数是
根据裴蜀定理,可以得知,如果
因此,我们只需要判断在数组 nums
中是否存在 nums
存在 nums
中的所有数的最大公约数也为
所以我们将题目转化为:判断数组 nums
中的所有数的最大公约数是否为 nums
,求出数组 nums
中的所有数的最大公约数即可。
时间复杂度 nums
的长度,而 nums
中的最大值。
class Solution:
def isGoodArray(self, nums: List[int]) -> bool:
return reduce(gcd, nums) == 1
class Solution {
public boolean isGoodArray(int[] nums) {
int g = 0;
for (int x : nums) {
g = gcd(x, g);
}
return g == 1;
}
private int gcd(int a, int b) {
return b == 0 ? a : gcd(b, a % b);
}
}
class Solution {
public:
bool isGoodArray(vector<int>& nums) {
int g = 0;
for (int x : nums) {
g = gcd(x, g);
}
return g == 1;
}
};
func isGoodArray(nums []int) bool {
g := 0
for _, x := range nums {
g = gcd(x, g)
}
return g == 1
}
func gcd(a, b int) int {
if b == 0 {
return a
}
return gcd(b, a%b)
}