Leetcode每日一练--47

3346. 执行操作后元素的最高频率 I

难度:中等

提示

给你一个整数数组 nums 和两个整数 knumOperations

你必须对 nums 执行 操作 numOperations 次。每次操作中,你可以:

  • 选择一个下标 i ,它在之前的操作中 没有 被选择过。
  • nums[i] 增加范围 [-k, k] 中的一个整数。

在执行完所有操作以后,请你返回 nums 中出现 频率最高 元素的出现次数。

一个元素 x频率 指的是它在数组中出现的次数。

示例 1:

**输入:**nums = [1,4,5], k = 1, numOperations = 2

**输出:**2

解释:

通过以下操作得到最高频率 2 :

  • nums[1] 增加 0 ,nums 变为 [1, 4, 5]
  • nums[2] 增加 -1 ,nums 变为 [1, 4, 4]

示例 2:

**输入:**nums = [5,11,20,20], k = 5, numOperations = 1

**输出:**2

解释:

通过以下操作得到最高频率 2 :

  • nums[1] 增加 0 。

提示:

  • 1 <= nums.length <= 10^5
  • 1 <= nums[i] <= 10^5
  • 0 <= k <= 10^5
  • 0 <= numOperations <= nums.length

代码

cpp 复制代码
#include <stdlib.h>
#include <string.h>

int compare(const void *a, const void *b) {
    return (*(int *)a - *(int *)b);
}

int lower_bound(int *arr, int size, int target) {
    int left = 0, right = size;
    while (left < right) {
        int mid = left + (right - left) / 2;
        if (arr[mid] < target) left = mid + 1;
        else right = mid;
    }
    return left;
}

int upper_bound(int *arr, int size, int target) {
    int left = 0, right = size;
    while (left < right) {
        int mid = left + (right - left) / 2;
        if (arr[mid] <= target) left = mid + 1;
        else right = mid;
    }
    return left;
}

int maxFrequency(int* nums, int numsSize, int k, int numOperations) {
    if (numsSize == 0) return 0;
    
    // 排序原始数组
    int *sorted = (int *)malloc(numsSize * sizeof(int));
    memcpy(sorted, nums, numsSize * sizeof(int));
    qsort(sorted, numsSize, sizeof(int), compare);
    
    // 统计原始频率 (值域: 1~100000)
    int *freq = (int *)calloc(100001, sizeof(int));
    for (int i = 0; i < numsSize; i++) {
        if (nums[i] >= 1 && nums[i] <= 100000) freq[nums[i]]++;
    }
    
    // 生成候选值 v = {x, x-k, x+k}
    int *candidates = (int *)malloc(3 * numsSize * sizeof(int));
    int idx = 0;
    for (int i = 0; i < numsSize; i++) {
        candidates[idx++] = nums[i];
        candidates[idx++] = nums[i] - k;
        candidates[idx++] = nums[i] + k;
    }
    
    // 排序并去重候选值
    int totalCandidates = idx;
    qsort(candidates, totalCandidates, sizeof(int), compare);
    int uniqueCandidates = 0;
    for (int i = 0; i < totalCandidates; i++) {
        if (i == 0 || candidates[i] != candidates[i - 1]) {
            candidates[uniqueCandidates++] = candidates[i];
        }
    }
    
    int best = 0;
    for (int i = 0; i < uniqueCandidates; i++) {
        int v = candidates[i];
        int low = v - k;
        int high = v + k;
        
        // 计算区间 [v-k, v+k] 内的元素数量
        int left_idx = lower_bound(sorted, numsSize, low);
        int right_idx = upper_bound(sorted, numsSize, high);
        int C_v = right_idx - left_idx;
        
        int candidate;
        if (v >= 1 && v <= 100000 && freq[v] > 0) {
            candidate = (C_v < freq[v] + numOperations) ? C_v : (freq[v] + numOperations);
        } else {
            candidate = (C_v < numOperations) ? C_v : numOperations;
        }
        
        if (candidate > best) best = candidate;
    }
    
    free(sorted);
    free(freq);
    free(candidates);
    return best;
}
相关推荐
Wect30 分钟前
LeetCode 5. 最长回文子串:DP + 中心扩展
前端·算法·typescript
糖果店的幽灵33 分钟前
决策树详解与sklearn实战
算法·决策树·sklearn
Lewiis36 分钟前
趣谈排序算法
算法·排序算法
ComputerInBook1 小时前
数字图像处理(4版)——第 8 章——图像压缩与水印(上)(Rafael C.Gonzalez&Richard E. Woods)
人工智能·算法·计算机视觉·图像压缩·图像水印
刀法如飞1 小时前
Python列表去重:从新手三连到高阶特技,20种解法全收录
python·算法·编程语言
minji...1 小时前
算法题 动态规划
算法·动态规划
水蓝烟雨2 小时前
3337. 字符串转换后的长度 II
算法·leetcode
MegaDataFlowers2 小时前
SiliconCompiler workflow
算法
_日拱一卒2 小时前
LeetCode:226翻转二叉树
数据结构·算法·leetcode
踩坑记录2 小时前
leetcode hot100 64. 最小路径和 medium 递归优化
leetcode·深度优先