【AcWing每日一题】4366. 上课睡觉

news2025/1/12 3:44:19

有 N 堆石子,每堆的石子数量分别为 a1,a2,…,aN。

你可以对石子堆进行合并操作,将两个相邻的石子堆合并为一个石子堆,例如,如果 a=[1,2,3,4,5],合并第 2,3 堆石子,则石子堆集合变为 a=[1,5,4,5]。

我们希望通过尽可能少的操作,使得石子堆集合中的每堆石子的数量都相同。

请你输出所需的最少操作次数。

本题一定有解,因为可以将所有石子堆合并为一堆。

输入格式
第一行包含整数 T,表示共有 T 组测试数据。
每组数据第一行包含整数 N。
第二行包含 N 个整数 a1,a2,…,aN。

输出格式
每组数据输出一行结果。

数据范围
1≤T≤10,
1≤N≤105,
0≤ai≤106,
∑ai≤106,
每个输入所有 N 之和不超过 105

自己的解法:

  • 堆排序的方法,对于每一组数据,建立初始小根堆,进行堆排序。
  • 每次排好一个元素后,进行对已排序序列检查,如果已经完成堆排序的序列的部分和(从前往后)可以大于等于目前的最大值,则合并这部分。(全是0的情况除外)
  • 合并后更新下次排序需要的参数,继续进行堆排序,合并。
  • 直至所有的数都一样或者合成一堆,算法停止,输出答案,进行下一组的堆排序。
  • 这需要每次合并后都检查是否每个数都一样。
  • 因为考虑到数据量的问题,用枚举是O(n2)的复杂度,会超时,用堆排序降到O(nlog2n)的复杂度。

我的推导过程:
在这里插入图片描述

但是考虑到如果每次排序后都进行检查,那么时间复杂度也会上升到O(n2)

自己的代码(没写完):

#include<iostream>
#include<vector>
using namespace std;

vector<vector<int> > v;	//v存放所有的数组 
int max_vj = -0xffff; 	//每组数据的最大值 
int res = 0;

//小根堆的建立
void HeapAdjust(vector<int> &A, int k, int len){
	int t = A[k];							//暂存这个分支结点 
	for(int i = 2*k; i <= len; i *= 2){		//从这个分支结点开始向下调整 
		if(i < len && A[i] > A[i+1]) i++;	//右孩子更小 
		if(t <= A[i]) break;				//分支结点已是子堆中的最小值,符合特性 
		else{								//不符合特性,需要调整 
			A[k] = A[i];					//换的时候只是覆盖
			k = i;							//下标要交换,下次还说与A[0]比较 
		}
	}
	A[k] = t;								//放到最终符合特性的位置上 
}
void BuildMinHeap(vector<int> &A, int len){
	//从最后一个分支结点开始,逐级向上建堆 
	for(int i = len/2; i > 0; i--)
		HeapAdjust(A, i, len);
} 
//堆排序
void HeapSort(vector<int> &A, int len){	 
	for(int i = len; i > 1; i--){	//n-1趟交换和建堆过程 
		swap(A[i], A[1]);			//堆顶元素和堆底元素互换
		//交换后检查是否符合合并的条件 
		//如果已经完成堆排序的序列的部分和(从后往前)可以大于等于目前的最大值,则合并这部分
		//合并后更新下一次需要的参数 
		if(){
			A[len-1] = A[len]+A[len-1];					//合并 
			if(A[len-1] > max_vj) max_vj = A[len-1];	//更新最大值、长度、合并次数 
			A[0]--;	len--; res++;
		}
		HeapAdjust(A, 1, i-1);		//将剩余的元素调整 
	}
}

int main(){
	int T, N, an;		//T组测试数据,每组数据包含N个整数an 
	cin >> T;			//测试数据的数量 
	vector<int> vj;		//每组数据存放的数组 
	//下面搞定输入 
	for(int i = 0; i < T; i++){
		cin >> N;		//N个整数 
		vj.clear();		//对数组先清空 
		vj.push_back(N);			//每个数组的第一个位置存放这个数组的长度 
		for(int j = 0; j < N; j++){
			cin >> an;	//输入a1~an 
			vj.push_back(an);			 
		}
		v.push_back(vj);//将每组数据放入v 
	}
	for(int i = 0; i < T; i++){
		vector<int> vj = v[i];
		if(vj[0] == 2 && vj[1] != vj[2]){	//如果只有两个数且不相等 
			cout << 1 << endl;
			continue;
		}else if(vj[0] == 1){				//如果只有一个数 
			cout << 0 << endl;
			continue;
		}
		//找出每组数据的最大值
		for(int j = 1; j <= vj[0]; j++)
			if(vj[j] > max_vj) max_vj = vj[j];
		//对于每组数据建立初始的小根堆
		BuildMinHeap(vj, vj[0]);			//建堆
		HeapSort(vj, vj[0]);				//堆排序的同时合并 
		cout << res << endl;
		max_vj = -0xffff;					//重置 
		res = 0;		
		
		//测试堆排序结果 
//		for(int i = 1; i <= vj[0]; i++) cout << vj[i] << " ";
//		cout << endl;
//		cout << max_vj << endl; 		
	} 	
	
	return 0;
}

y总的方法:
他的方法是用的数论的推导
总的石子个数sum,最终每堆石子cnt个,cnt只有整除sum才可以
cnt的数量等于sum的约数个数,平均每个数的约数个数是O(logn)个

106以内,约数个数最多的数是720720,有240个约数
int范围内,约数个数最多的数有1600个

  • 知道了这个前提,我们就可以用枚举的方法,枚举所有的约数找到可以取到的并整除sum的数
  • 在所有的数里面找到合并次数最少的方案
  • 我们知道,最终合并后有sum/cnt堆石子,原来有n堆
  • 所以合并的次数为 n-sum/cnt
  • 我们只需要找到一个最小的符合条件的cnt
  • 所以我们只需要从小到大枚举cnt,看是否成立就可以了

一开始还有一个条件我没注意到,只能合并相邻的两堆,这样堆排序好像没法用了

那我们如何判断cnt是否成立?(能否将每一堆的数量变成cnt)

  • 对于每个cnt,我们对原始序列从前向后枚举
  • 如果当前堆的石子数量大于cnt,那一定无解,枚举下一个cnt
  • 如果当前堆的石子数量等于cnt,如果后面的数量是0,可以加到前面或后面的一堆;如果不是0,那就合并到下一段
  • 根据计算,大约240个约数,最多用107次计算量,可以在规定的时间内完成

最后附一下代码:

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 100010;

int n;
int w[N];

//判断当前的约数是否符合条件
bool check(int cnt)
{
	//从前向后枚举序列的每一段
    for (int i = 0, s = 0; i < n; i ++ )
    {
        s += w[i];					//合并相邻的两段
        if (s > cnt) return false;	//如果合并后的石子数量大于cnt,无解
        if (s == cnt) s = 0;		//如果等于这个约数cnt,继续合并后面的段
    }
    return true;
}

int main()
{
    int T;
    scanf("%d", &T);
    while (T -- )
    {
        scanf("%d", &n);				//石子的堆数
        int sum = 0;					//石子的总数sum
        for (int i = 0; i < n; i ++ )	//输入每堆石子个数
        {	
            scanf("%d", &w[i]);		
            sum += w[i];				//计入到总数sum
        }

        for (int i = n; i; i -- )		//对于每堆石子,枚举寻找符合条件的cnt
            if (sum % i == 0 && check(sum / i))//如果cnt能够整除sum,并且操作次数最小,且有解
            {
                printf("%d\n", n - i);	//输出答案
                break;					
            }
    }
    return 0;
}

//作者:yxc
//链接:https://www.acwing.com/activity/content/code/content/5027948/
//来源:AcWing
//著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/127266.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

LeetCodeday02

977.有序数组的平方 给你一个按 非递减顺序 排序的整数数组 nums&#xff0c;返回 每个数字的平方 组成的新数组&#xff0c;要求也按 非递减顺序 排序。 示例 1&#xff1a; 输入&#xff1a;nums [-4,-1,0,3,10] 输出&#xff1a;[0,1,9,16,100] 解释&#xff1a;平方后&am…

Java开发学习(三十五)----SpringBoot快速入门及起步依赖解析

一、SpringBoot简介 SpringBoot 是由 Pivotal 团队提供的全新框架&#xff0c;其设计目的是用来简化 Spring 应用的初始搭建以及开发过程。 使用了 Spring 框架后已经简化了我们的开发。而 SpringBoot 又是对 Spring 开发进行简化的&#xff0c;可想而知 SpringBoot 使用的简…

做报表要用什么插件?

Excel 作为大家最熟悉的报表工具&#xff0c;很多表哥表姐每天都在使用&#xff0c;为了加强 Excel 的报表功能&#xff0c;市面上有非常多的 Excel 增强插件&#xff0c;为 Excel 增加了千奇百怪的能力。今天给大家介绍一款专门用来做中国式复杂报表的Excel 插件&#xff1a;思…

【不一样的递归大法】

&#x1f381;递归&#x1f385;递归&#x1f98c;定义&#x1f385;何时用递归&#xff1a;递归三板斧&#x1f98c;递归递归&#x1f98c;递归大法&#xff1a;三板斧&#x1f385;如何快速写出递归函数&#xff1a;宏观的角度&#x1f385;解题突破&#x1f98c;整数序列相关…

一文了解什么是NFT

一、什么是NFT NFT 是我们可以用来代表独特物品所有权的代币。他们让我们对艺术品、收藏品甚至房地产等事物进行代币化。资产的所有权由以太坊区块链保护——没有人可以修改所有权记录或复制/粘贴新的 NFT。 NFT 代表不可替代的代币。Non-fungible 是一个经济学术语&#xff…

伪操作和混合汇编

目录 一、伪操作: 二、C和汇编的混合编程 三、ATPCS协议(ARM-THUMB Procedure Call Standard) 一、伪操作: 不会生成代码&#xff0c;只是在编译之前告诉编译器怎么编译 GNU的伪操作一般都以‘.’开头 .global symbol 将symbo…

WEB 安全,浅谈 XSS 攻击(附简单实例)

什么是 XSS XSS(Cross-Site-Scripting)&#xff0c;跨站脚本攻击&#xff0c;因为缩写和 CSS 重叠&#xff0c;被别人抢先了&#xff0c;所以只能叫做 XSS。 攻击者可以利用这种漏洞在网站上注入恶意的客户端代码。若受害者运行这些恶意代码&#xff0c;攻击者就可以突破网站…

电感和磁珠有哪些区别?

由于电感和磁珠&#xff0c;很多人会容易认错&#xff0c;本期内容就讲讲&#xff0c;有哪些相似之处&#xff01; 磁珠与电感不仅在外形上相似&#xff0c;他们在功能上也存在很多相似之处&#xff0c;甚至有时候磁珠和电感可以相互代替。但是磁珠与电感也不完全等同&#xff…

开关电源环路稳定性分析(09)——环路补偿六步法

大家好&#xff0c;这里是大话硬件。 我们来回顾一下前面8讲的内容&#xff0c;主要对下面的知识点进行了分析&#xff1a; 系统框图 反馈环节传递函数 功率级传递函数 PWM级传递函数 传递函数计算 如果我们把开关电源看成是不同的电路模块拼接而成&#xff0c;现在已经知…

c++模板认识以及使用

我们都知道c有函数重载的概念&#xff0c;比如我们写一个相加的函数&#xff0c;以整数为例&#xff0c;我们大概率是这样写&#xff1a; int Add(int x,int y) {return xy; } 并且我们知道c函数重载的概念&#xff0c;于是我们若是想写double类型&#xff0c;float类型&…

centos7安装php7.1 验证码GD库扩展

php安装不推荐使用源码包安装&#xff0c;版本太多。线上从php5.x一直升级到php7.1 程序可兼容 配置yum源 rpm -Uvh https://dl.fedoraproject.org/pub/epel/epel-release-latest-7.noarch.rpm rpm -Uvh https://mirror.webtatic.com/yum/el7/webtatic-release.rpm 添加epel…

使用Stabel Diffusion

Stabel Diffusion是由CompVis、stabel AI和LAION的研究人员和工程师创建的文本到图像的潜在扩散模型。它由来自LAION-5B数据库子集的512x512图像进行训练。LAION-5B是目前最大的、可自由访问的多模态数据集。 在这篇文章中&#xff0c;将介绍如何使用diffusion库实现Stabel Di…

MAC安装jmeter以及JDK配置

JDK配置 一、检查是否安装了jdk 打开终端&#xff0c;输入java -version校验jdk是否安装 我这已经安装了版本1.8.0版本的 若没有安装&#xff0c;则去官网下载jdk并安装 1、jdk下载官网&#xff1a;Java Downloads | Oracle tar包或者dmg&#xff0c;二者区别在于&#xff1a;…

7、GPIO输入按键检测(外部中断)

目录 0x01、简介 0x001、EXTI 简介 0x002、EXTI 功能框图 0x003、中断/事件线 0x02、硬件设计 0x03、相关库函数 0x0001、外部中断初始化 0x0002、外部中断GPIO引脚选择 0x04、编写函数 0x001、按键外部中断初始化 0x002、中断函数 0x05、源程序下载地址 0x01、简介…

Android插件化换肤原理—— 布局加载过程、View创建流程、Resources 浅析

前言 继上次 WebView 干货分享后&#xff0c;本次将分享下自己在探索学习 App 换肤功能过程中的相关知识&#xff0c;着重分享换肤的原理以及实现思路。 由于篇幅原因分为两篇博客&#xff0c;本文主要分析了 Android 布局加载流程&#xff0c;下一篇将具体讲解插件化换肤实现…

Linux——标准IO

文件的基础 概念&#xff1a;一组相关数据的有序集合 文件的类型&#xff1a; 常规文件-r 目录文件-d 字符设备文件-c&#xff1a;键盘 块设备文件-b:U盘 磁盘 管道文件-p 套接字文件-s 符号链接文件-I&#xff1a;快捷方式 标准I/O 流 file 标准IO用一个结构体类型来保存打…

数据质量管理—理论大纲与实践(B站)

0、背景 故事的开头&#xff0c;是一位业务部门的同事找到我们&#xff0c;咨询了一个经典问题&#xff1a; 「需求方经常说我们做的报表看起来数据不准&#xff0c;有什么办法吗&#xff1f;」 为了解释这个问题&#xff0c;我以我们团队在数据质量管理中积累下来的方法&am…

决策树和随机森林的python实现

文章目录决策树实现方法测试更好地展示结果调参调整max_depthscoring利用GridSearchCV确定最佳max_depthmin_samples_splitmin_impurity_decreasemax_features多参数同时选优采用最优参数特征重要性排序随机森林测试调参n_estimators调整max_depth调整max_features调整min_samp…

黑马Hive+Spark离线数仓工业项目--数仓事实层DWB层构建(2)

工单事实指标构建 目标&#xff1a;实现DWB层工单事实指标表的构建 实施 建表 抽取 安装事实指标需求分析 目标&#xff1a;掌握DWB层安装事实指标表的需求分析 路径 - step1&#xff1a;目标需求 - step2&#xff1a;数据来源 实施 目标需求&#xff1a;基于设备安装信…

Python【继承】复写使用父类成员

继承&#xff1a;继承就是一个类&#xff0c;继承另外一个类的成员变量&#xff08;属性&#xff09;和成员方法 继承的作用&#xff1a;子类通过继承父类的属性和方法&#xff0c;在调用的时候&#xff0c;除了可以使用子类自身的成员方法和属性外&#xff0c;还可以使用父类…