comments | difficulty | edit_url | rating | source | tags | |||||
---|---|---|---|---|---|---|---|---|---|---|
true |
困难 |
2561 |
第 288 场周赛 Q4 |
|
Alice 是 n
个花园的园丁,她想通过种花,最大化她所有花园的总美丽值。
给你一个下标从 0 开始大小为 n
的整数数组 flowers
,其中 flowers[i]
是第 i
个花园里已经种的花的数目。已经种了的花 不能 移走。同时给你 newFlowers
,表示 Alice 额外可以种花的 最大数目 。同时给你的还有整数 target
,full
和 partial
。
如果一个花园有 至少 target
朵花,那么这个花园称为 完善的 ,花园的 总美丽值 为以下分数之 和 :
- 完善 花园数目乘以
full
. - 剩余 不完善 花园里,花的 最少数目 乘以
partial
。如果没有不完善花园,那么这一部分的值为0
。
请你返回 Alice 种最多 newFlowers
朵花以后,能得到的 最大 总美丽值。
示例 1:
输入:flowers = [1,3,1,1], newFlowers = 7, target = 6, full = 12, partial = 1 输出:14 解释:Alice 可以按以下方案种花 - 在第 0 个花园种 2 朵花 - 在第 1 个花园种 3 朵花 - 在第 2 个花园种 1 朵花 - 在第 3 个花园种 1 朵花 花园里花的数目为 [3,6,2,2] 。总共种了 2 + 3 + 1 + 1 = 7 朵花。 只有 1 个花园是完善的。 不完善花园里花的最少数目是 2 。 所以总美丽值为 1 * 12 + 2 * 1 = 12 + 2 = 14 。 没有其他方案可以让花园总美丽值超过 14 。
示例 2:
输入:flowers = [2,4,5,3], newFlowers = 10, target = 5, full = 2, partial = 6 输出:30 解释:Alice 可以按以下方案种花 - 在第 0 个花园种 3 朵花 - 在第 1 个花园种 0 朵花 - 在第 2 个花园种 0 朵花 - 在第 3 个花园种 2 朵花 花园里花的数目为 [5,4,5,5] 。总共种了 3 + 0 + 0 + 2 = 5 朵花。 有 3 个花园是完善的。 不完善花园里花的最少数目为 4 。 所以总美丽值为 3 * 2 + 4 * 6 = 6 + 24 = 30 。 没有其他方案可以让花园总美丽值超过 30 。 注意,Alice可以让所有花园都变成完善的,但这样她的总美丽值反而更小。
提示:
1 <= flowers.length <= 105
1 <= flowers[i], target <= 105
1 <= newFlowers <= 1010
1 <= full, partial <= 105
我们注意到,如果一个花园中种的花的数目已经大于等于
我们不妨枚举有多少个花园最终成为完善的花园,假设初始时有
接下来,我们枚举完善花园的数目
我们更新剩余可种植的花
否则,我们在
时间复杂度
class Solution:
def maximumBeauty(
self, flowers: List[int], newFlowers: int, target: int, full: int, partial: int
) -> int:
flowers.sort()
n = len(flowers)
s = list(accumulate(flowers, initial=0))
ans, i = 0, n - bisect_left(flowers, target)
for x in range(i, n + 1):
newFlowers -= 0 if x == 0 else max(target - flowers[n - x], 0)
if newFlowers < 0:
break
l, r = 0, n - x - 1
while l < r:
mid = (l + r + 1) >> 1
if flowers[mid] * (mid + 1) - s[mid + 1] <= newFlowers:
l = mid
else:
r = mid - 1
y = 0
if r != -1:
cost = flowers[l] * (l + 1) - s[l + 1]
y = min(flowers[l] + (newFlowers - cost) // (l + 1), target - 1)
ans = max(ans, x * full + y * partial)
return ans
class Solution {
public long maximumBeauty(int[] flowers, long newFlowers, int target, int full, int partial) {
Arrays.sort(flowers);
int n = flowers.length;
long[] s = new long[n + 1];
for (int i = 0; i < n; ++i) {
s[i + 1] = s[i] + flowers[i];
}
long ans = 0;
int x = 0;
for (int v : flowers) {
if (v >= target) {
++x;
}
}
for (; x <= n; ++x) {
newFlowers -= (x == 0 ? 0 : Math.max(target - flowers[n - x], 0));
if (newFlowers < 0) {
break;
}
int l = 0, r = n - x - 1;
while (l < r) {
int mid = (l + r + 1) >> 1;
if ((long) flowers[mid] * (mid + 1) - s[mid + 1] <= newFlowers) {
l = mid;
} else {
r = mid - 1;
}
}
long y = 0;
if (r != -1) {
long cost = (long) flowers[l] * (l + 1) - s[l + 1];
y = Math.min(flowers[l] + (newFlowers - cost) / (l + 1), target - 1);
}
ans = Math.max(ans, (long) x * full + y * partial);
}
return ans;
}
}
class Solution {
public:
long long maximumBeauty(vector<int>& flowers, long long newFlowers, int target, int full, int partial) {
sort(flowers.begin(), flowers.end());
int n = flowers.size();
long long s[n + 1];
s[0] = 0;
for (int i = 1; i <= n; ++i) {
s[i] = s[i - 1] + flowers[i - 1];
}
long long ans = 0;
int i = flowers.end() - lower_bound(flowers.begin(), flowers.end(), target);
for (int x = i; x <= n; ++x) {
newFlowers -= (x == 0 ? 0 : max(target - flowers[n - x], 0));
if (newFlowers < 0) {
break;
}
int l = 0, r = n - x - 1;
while (l < r) {
int mid = (l + r + 1) >> 1;
if (1LL * flowers[mid] * (mid + 1) - s[mid + 1] <= newFlowers) {
l = mid;
} else {
r = mid - 1;
}
}
int y = 0;
if (r != -1) {
long long cost = 1LL * flowers[l] * (l + 1) - s[l + 1];
long long mx = flowers[l] + (newFlowers - cost) / (l + 1);
long long threshold = target - 1;
y = min(mx, threshold);
}
ans = max(ans, 1LL * x * full + 1LL * y * partial);
}
return ans;
}
};
func maximumBeauty(flowers []int, newFlowers int64, target int, full int, partial int) int64 {
sort.Ints(flowers)
n := len(flowers)
s := make([]int, n+1)
for i, x := range flowers {
s[i+1] = s[i] + x
}
ans := 0
i := n - sort.SearchInts(flowers, target)
for x := i; x <= n; x++ {
if x > 0 {
newFlowers -= int64(max(target-flowers[n-x], 0))
}
if newFlowers < 0 {
break
}
l, r := 0, n-x-1
for l < r {
mid := (l + r + 1) >> 1
if int64(flowers[mid]*(mid+1)-s[mid+1]) <= newFlowers {
l = mid
} else {
r = mid - 1
}
}
y := 0
if r != -1 {
cost := flowers[l]*(l+1) - s[l+1]
y = min(flowers[l]+int((newFlowers-int64(cost))/int64(l+1)), target-1)
}
ans = max(ans, x*full+y*partial)
}
return int64(ans)
}
function maximumBeauty(
flowers: number[],
newFlowers: number,
target: number,
full: number,
partial: number,
): number {
flowers.sort((a, b) => a - b);
const n = flowers.length;
const s: number[] = Array(n + 1).fill(0);
for (let i = 1; i <= n; i++) {
s[i] = s[i - 1] + flowers[i - 1];
}
let x = flowers.filter(f => f >= target).length;
let ans = 0;
for (; x <= n; ++x) {
newFlowers -= x === 0 ? 0 : Math.max(target - flowers[n - x], 0);
if (newFlowers < 0) {
break;
}
let l = 0;
let r = n - x - 1;
while (l < r) {
const mid = (l + r + 1) >> 1;
if (flowers[mid] * (mid + 1) - s[mid + 1] <= newFlowers) {
l = mid;
} else {
r = mid - 1;
}
}
let y = 0;
if (r !== -1) {
const cost = flowers[l] * (l + 1) - s[l + 1];
y = Math.min(flowers[l] + Math.floor((newFlowers - cost) / (l + 1)), target - 1);
}
ans = Math.max(ans, x * full + y * partial);
}
return ans;
}