每日一题(二分+枚举)

1.题目解析

分组 (nowcoder.com)

主要题意就是,按照不同的序号进行分组,同一序号只能放在同一组,并且要求答案是分成要求的组数且在符合组数的前提下求出最大值最小的一个。

2.思路分析

这个题可以说是一个比较容易误导人的一道题,永远都是想通过人数来推组数,这样复杂度也很高重复计算也很多,这题主要解法是一种反推法:枚举最终的结果,从来推出最多人数组的最小数。

1.统计人数

使用hash表进行统计,声部为键,出现的次数为值

2.枚举最终结果,找出符合m组且最大人数最小的

人数越多,组数越少,所以说枚举范围就是 1 ~ 最多声部的人数

比如枚举到3了,最多声部的人数为8,8就可以分成2个3 一个2,其他的也按照这样分比如 人数2 就是一组,这样会得到组数count 然后和m比较如果是符合的count <= m直接返回即可,因为如果是7也可以随便拆出来一组,因为他不管我们怎么返回,他要最大 人数的那个组的人数。

3.二分进行优化

存在二段性

下面是我手写的分析过程和模拟过程

3.代码实现

1.暴力枚举

java 复制代码
 public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        // 注意 hasNext 和 hasNextLine 的区别
        while (in.hasNextInt()) { // 注意 while 处理多个 case
            int n = in.nextInt();
            Map<Integer,Integer> hash = new HashMap<>();
            int m = in.nextInt();
            int max = 0;//所有声部的最大值
            for (int i = 0; i < n; i++) {
                int x = in.nextInt();
                hash.put(x,hash.getOrDefault(x,0) + 1);
                max = Math.max(hash.get(x), max);
            }
            //处理边界情况, 类型大于组数就无法分
            int kinds = hash.size();
            if (kinds > m){
                System.out.println(-1);
                continue;
            }
            for (int i = 1; i <= max; i++) {
                //判断最多人数为x的时候是否能分成m组
                int count = 0;//统计能分成组
                for (int value : hash.values()) {
                    count += value / i + (value % i == 0 ? 0 : 1);
                }
                if (count <= m) {
                    System.out.println(i);
                    break;
                }
            }
        }
    }

2.二分进行优化

java 复制代码
public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        // 注意 hasNext 和 hasNextLine 的区别
        while (in.hasNextInt()) { // 注意 while 处理多个 case
            int n = in.nextInt();
            Map<Integer,Integer> hash = new HashMap<>();
            int m = in.nextInt();
            int max = 0;//所有声部的最大值
            for (int i = 0; i < n; i++) {
                int x = in.nextInt();
                hash.put(x,hash.getOrDefault(x,0) + 1);
                max = Math.max(hash.get(x), max);
            }
            //处理边界情况, 类型大于组数就无法分
            int kinds = hash.size();
            if (kinds > m){
                System.out.println(-1);
                continue;
            }
            int ans = 1; // 记录答案
            int start = 1;
            int end = max;
            while(start <= end){ // 二分查找
                int mid = (start + end) / 2;
                int count = 0;//统计能分成多少组
                for (int value : hash.values()) {
                    count += value / mid + (value % mid == 0 ? 0 : 1);//最少组数
                }
                if (count <= m) {//是答案但不是最符合的往右搜
                    end = mid - 1;
                    ans = mid; // 更新答案
                }else {//不符合要求,组数太多了,往右搜
                    start = mid + 1;
                }
            }
            System.out.println(ans); // 输出答案
        }
    }

4.总结

通过解决这个问题,我深入理解了分组算法的原理和应用。我的解决方案虽然简单,但已经相当有效。在未来,我可以进一步探索和优化分组算法,以解决更复杂的问题,并应用于更广泛的领域。

相关推荐
shymoy17 分钟前
Radix Sorts
数据结构·算法·排序算法
风影小子26 分钟前
注册登录学生管理系统小项目
算法
黑龙江亿林等保28 分钟前
深入探索哈尔滨二级等保下的负载均衡SLB及其核心算法
运维·算法·负载均衡
lucy1530275107931 分钟前
【青牛科技】GC5931:工业风扇驱动芯片的卓越替代者
人工智能·科技·单片机·嵌入式硬件·算法·机器学习
杜杜的man1 小时前
【go从零单排】迭代器(Iterators)
开发语言·算法·golang
小沈熬夜秃头中୧⍤⃝1 小时前
【贪心算法】No.1---贪心算法(1)
算法·贪心算法
木向2 小时前
leetcode92:反转链表||
数据结构·c++·算法·leetcode·链表
阿阿越2 小时前
算法每日练 -- 双指针篇(持续更新中)
数据结构·c++·算法
skaiuijing2 小时前
Sparrow系列拓展篇:对调度层进行抽象并引入IPC机制信号量
c语言·算法·操作系统·调度算法·操作系统内核
Star Patrick2 小时前
算法训练(leetcode)二刷第十九天 | *39. 组合总和、*40. 组合总和 II、*131. 分割回文串
python·算法·leetcode