comments | difficulty | edit_url | rating | source | tags | ||||
---|---|---|---|---|---|---|---|---|---|
true |
困难 |
1808 |
第 80 场双周赛 Q4 |
|
一个数组的 分数 定义为数组之和 乘以 数组的长度。
- 比方说,
[1, 2, 3, 4, 5]
的分数为(1 + 2 + 3 + 4 + 5) * 5 = 75
。
给你一个正整数数组 nums
和一个整数 k
,请你返回 nums
中分数 严格小于 k
的 非空整数子数组数目。
子数组 是数组中的一个连续元素序列。
示例 1:
输入:nums = [2,1,4,3,5], k = 10 输出:6 解释: 有 6 个子数组的分数小于 10 : - [2] 分数为 2 * 1 = 2 。 - [1] 分数为 1 * 1 = 1 。 - [4] 分数为 4 * 1 = 4 。 - [3] 分数为 3 * 1 = 3 。 - [5] 分数为 5 * 1 = 5 。 - [2,1] 分数为 (2 + 1) * 2 = 6 。 注意,子数组 [1,4] 和 [4,3,5] 不符合要求,因为它们的分数分别为 10 和 36,但我们要求子数组的分数严格小于 10 。
示例 2:
输入:nums = [1,1,1], k = 5 输出:5 解释: 除了 [1,1,1] 以外每个子数组分数都小于 5 。 [1,1,1] 分数为 (1 + 1 + 1) * 3 = 9 ,大于 5 。 所以总共有 5 个子数组得分小于 5 。
提示:
1 <= nums.length <= 105
1 <= nums[i] <= 105
1 <= k <= 1015
我们先计算出数组
接下来,我们枚举数组
时间复杂度
class Solution:
def countSubarrays(self, nums: List[int], k: int) -> int:
s = list(accumulate(nums, initial=0))
ans = 0
for i in range(1, len(s)):
left, right = 0, i
while left < right:
mid = (left + right + 1) >> 1
if (s[i] - s[i - mid]) * mid < k:
left = mid
else:
right = mid - 1
ans += left
return ans
class Solution {
public long countSubarrays(int[] nums, long k) {
int n = nums.length;
long[] s = new long[n + 1];
for (int i = 0; i < n; ++i) {
s[i + 1] = s[i] + nums[i];
}
long ans = 0;
for (int i = 1; i <= n; ++i) {
int left = 0, right = i;
while (left < right) {
int mid = (left + right + 1) >> 1;
if ((s[i] - s[i - mid]) * mid < k) {
left = mid;
} else {
right = mid - 1;
}
}
ans += left;
}
return ans;
}
}
class Solution {
public:
long long countSubarrays(vector<int>& nums, long long k) {
int n = nums.size();
long long s[n + 1];
s[0] = 0;
for (int i = 0; i < n; ++i) {
s[i + 1] = s[i] + nums[i];
}
long long ans = 0;
for (int i = 1; i <= n; ++i) {
int left = 0, right = i;
while (left < right) {
int mid = (left + right + 1) >> 1;
if ((s[i] - s[i - mid]) * mid < k) {
left = mid;
} else {
right = mid - 1;
}
}
ans += left;
}
return ans;
}
};
func countSubarrays(nums []int, k int64) (ans int64) {
n := len(nums)
s := make([]int64, n+1)
for i, v := range nums {
s[i+1] = s[i] + int64(v)
}
for i := 1; i <= n; i++ {
left, right := 0, i
for left < right {
mid := (left + right + 1) >> 1
if (s[i]-s[i-mid])*int64(mid) < k {
left = mid
} else {
right = mid - 1
}
}
ans += int64(left)
}
return
}
我们可以使用双指针的方式,维护一个滑动窗口,使得窗口内的元素和小于
时间复杂度
class Solution:
def countSubarrays(self, nums: List[int], k: int) -> int:
ans = s = j = 0
for i, v in enumerate(nums):
s += v
while s * (i - j + 1) >= k:
s -= nums[j]
j += 1
ans += i - j + 1
return ans
class Solution {
public long countSubarrays(int[] nums, long k) {
long ans = 0, s = 0;
for (int i = 0, j = 0; i < nums.length; ++i) {
s += nums[i];
while (s * (i - j + 1) >= k) {
s -= nums[j++];
}
ans += i - j + 1;
}
return ans;
}
}
class Solution {
public:
long long countSubarrays(vector<int>& nums, long long k) {
long long ans = 0, s = 0;
for (int i = 0, j = 0; i < nums.size(); ++i) {
s += nums[i];
while (s * (i - j + 1) >= k) {
s -= nums[j++];
}
ans += i - j + 1;
}
return ans;
}
};
func countSubarrays(nums []int, k int64) (ans int64) {
s, j := 0, 0
for i, v := range nums {
s += v
for int64(s*(i-j+1)) >= k {
s -= nums[j]
j++
}
ans += int64(i - j + 1)
}
return
}