给你一个下标从 1 开始、由 n
个整数组成的数组。
如果一组数字中每对元素的乘积都是一个完全平方数,则称这组数字是一个 完全集 。
下标集 {1, 2, ..., n}
的子集可以表示为 {i1, i2, ..., ik}
,我们定义对应该子集的 元素和 为 nums[i1] + nums[i2] + ... + nums[ik]
。
返回下标集 {1, 2, ..., n}
的 完全子集 所能取到的 最大元素和 。
完全平方数是指可以表示为一个整数和其自身相乘的数。
示例 1:
输入:nums = [8,7,3,5,7,2,4,9] 输出:16 解释:除了由单个下标组成的子集之外,还有两个下标集的完全子集:{1,4} 和 {2,8} 。 与下标 1 和 4 对应的元素和等于 nums[1] + nums[4] = 8 + 5 = 13 。 与下标 2 和 8 对应的元素和等于 nums[2] + nums[8] = 7 + 9 = 16 。 因此,下标集的完全子集可以取到的最大元素和为 16 。
示例 2:
输入:nums = [5,10,3,10,1,13,7,9,4] 输出:19 解释:除了由单个下标组成的子集之外,还有四个下标集的完全子集:{1,4}、{1,9}、{2,8}、{4,9} 和 {1,4,9} 。 与下标 1 和 4 对应的元素和等于 nums[1] + nums[4] = 5 + 10 = 15 。 与下标 1 和 9 对应的元素和等于 nums[1] + nums[9] = 5 + 4 = 9 。 与下标 2 和 8 对应的元素和等于 nums[2] + nums[8] = 10 + 9 = 19 。 与下标 4 和 9 对应的元素和等于 nums[4] + nums[9] = 10 + 4 = 14 。 与下标 1、4 和 9 对应的元素和等于 nums[1] + nums[4] + nums[9] = 5 + 10 + 4 = 19 。 因此,下标集的完全子集可以取到的最大元素和为 19 。
提示:
1 <= n == nums.length <= 104
1 <= nums[i] <= 109
方法一:枚举
我们注意到,如果一个数字可以表示成
因此,我们可以在
最后返回答案
时间复杂度
class Solution:
def maximumSum(self, nums: List[int]) -> int:
n = len(nums)
ans = 0
for k in range(1, n + 1):
t = 0
j = 1
while k * j * j <= n:
t += nums[k * j * j - 1]
j += 1
ans = max(ans, t)
return ans
class Solution {
public long maximumSum(List<Integer> nums) {
long ans = 0;
int n = nums.size();
boolean[] used = new boolean[n + 1];
int bound = (int) Math.floor(Math.sqrt(n));
int[] squares = new int[bound + 1];
for (int i = 1; i <= bound + 1; i++) {
squares[i - 1] = i * i;
}
for (int i = 1; i <= n; i++) {
long res = 0;
int idx = 0;
int curr = i * squares[idx];
while (curr <= n) {
res += nums.get(curr - 1);
curr = i * squares[++idx];
}
ans = Math.max(ans, res);
}
return ans;
}
}
class Solution {
public long maximumSum(List<Integer> nums) {
long ans = 0;
int n = nums.size();
for (int k = 1; k <= n; ++k) {
long t = 0;
for (int j = 1; k * j * j <= n; ++j) {
t += nums.get(k * j * j - 1);
}
ans = Math.max(ans, t);
}
return ans;
}
}
class Solution {
public:
long long maximumSum(vector<int>& nums) {
long long ans = 0;
int n = nums.size();
for (int k = 1; k <= n; ++k) {
long long t = 0;
for (int j = 1; k * j * j <= n; ++j) {
t += nums[k * j * j - 1];
}
ans = max(ans, t);
}
return ans;
}
};
func maximumSum(nums []int) (ans int64) {
n := len(nums)
for k := 1; k <= n; k++ {
var t int64
for j := 1; k*j*j <= n; j++ {
t += int64(nums[k*j*j-1])
}
ans = max(ans, t)
}
return
}
function maximumSum(nums: number[]): number {
let ans = 0;
const n = nums.length;
for (let k = 1; k <= n; ++k) {
let t = 0;
for (let j = 1; k * j * j <= n; ++j) {
t += nums[k * j * j - 1];
}
ans = Math.max(ans, t);
}
return ans;
}