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;
}
相关推荐
2501_941236212 小时前
C++与Node.js集成
开发语言·c++·算法
晨非辰2 小时前
【数据结构初阶系列】归并排序全透视:从算法原理全分析到源码实战应用
运维·c语言·数据结构·c++·人工智能·python·深度学习
云边有个稻草人6 小时前
部分移动(Partial Move)的使用场景:Rust 所有权拆分的精细化实践
开发语言·算法·rust
泡沫冰@7 小时前
数据结构(20)
数据结构
松涛和鸣7 小时前
11.C 语言学习:递归、宏定义、预处理、汉诺塔、Fibonacci 等
linux·c语言·开发语言·学习·算法·排序算法
2501_941111248 小时前
C++与自动驾驶系统
开发语言·c++·算法
2501_941111699 小时前
C++中的枚举类高级用法
开发语言·c++·算法
jz_ddk9 小时前
[算法] 算法PK:LMS与RLS的对比研究
人工智能·神经网络·算法·信号处理·lms·rls·自适应滤波
Miraitowa_cheems9 小时前
LeetCode算法日记 - Day 106: 两个字符串的最小ASCII删除和
java·数据结构·算法·leetcode·深度优先