牛客题解 | 剪绳子(进阶版)

news/2025/2/25 19:19:09/文章来源:https://www.cnblogs.com/wc529065/p/18737047

题目

题目链接

题目的主要信息:
  • 把一根长度为\(n\)的绳子分成\(m\)段,每段长度都是整数
  • 求每段长度乘积的最大值
  • 由于答案过大,请对 998244353 取模
举一反三:

学习完本题的思路你可以解决如下题目:

JZ14. 剪绳子

方法:快速幂+快速乘法(推荐使用)

知识点1:贪心思想

贪心思想属于动态规划思想中的一种,其基本原理是找出整体当中给的每个局部子结构的最优解,并且最终将所有的这些局部最优解结合起来形成整体上的一个最优解。

知识点2:分治:

分治即“分而治之”,“分”指的是将一个大而复杂的问题划分成多个性质相同但是规模更小的子问题,子问题继续按照这样划分,直到问题可以被轻易解决;“治”指的是将子问题单独进行处理。经过分治后的子问题,需要将解进行合并才能得到原问题的解,因此整个分治过程经常用递归来实现。

思路:

根据均值不等式,有:\(\frac{n_1+n_2+...+n_m}{m}\geq \sqrt[m]{n_1n_2...n_m}\),等号当且仅当\(n_1=n_2=n_3=...n_m\)时成立,因为加法部分和是固定的绳子总长度,因此要使乘积最大,应该以相等的长度等分成多段。

如果将绳子按照每段长度为\(x\)等分成\(m\)段,则\(n=mx\),乘积为\(x^m\),因为有\(x^m=x^{\frac{n}{x}}=(x^{\frac{1}{x}})^n\)因此当\(x^{\frac{1}{x}}\)取最大值时取最大值。

\(y=x^{\frac{1}{x}}\),即求这个函数的极值即可直到绳子等分成长度为多少可以使乘积最大。根据取对数、求导、求极值等一系列数学操作,得驻点为\(x_0=e\),即极大值需要将绳子分成每段e,但是绳子长度只能是整数,靠近e的只有2
和3,二者分别代入公式,发现当\(x=3\)是,乘积达到最大。

因此后续,使用贪心思想,不断将绳子分成每段长度为3即可,不足3的可以考虑,如果最后剩余的是2,直接乘上,如果最后剩余的是1,则取出一个3组成4分成长度为2的两段,因为\(2*2>1*3\)

具体做法:

  • step 1:将问题分成三种情况,使用快速幂和快速乘法直接计算幂。
  • step 2:n整除3的时候,即可以全部完成分成长度为3的小段,一共\(n/3\)段,计算\(3^{n/3}\)即可。
  • step 3:n除3余1的时候,需要拿出一个3个1组合称,一共\(n/3-1\)段长度为3的,2段长度为2的,计算\(2*2*3^{n/3-1}\)即可;
  • step 4:n除3余2的时候,直接将剩下长度为2的段乘在之前的乘积上,计算\(2*3^{n/3}\)即可。

计算幂为了缩短时间,采用快速幂加快速乘法优化:

快速幂:如果我们要计算\(5^{10}\),常规的算法是\(5*5=25\),然后再\(25*5=125\),如此往下,一共是\(9\)次运算,即\(n-1\)次。但是我们可以考虑这样:\(5*5=25\)(二次)、\(25*25=625\)(四次)、\(625*625=...\)(八次),这是一个二分的思维,运算次数缩减到了\(log_2n\)次,公式如下:

alt

快速乘法:直接计算\(x^a\)会超出long的表示范围,因此我们可以考虑用加法来代替乘法,并在这其中取模。就比如\(a*b=a*(b_1+b_2+b_3+...)\),其中\(b_i\)是数字\(b\)的二进制各位,假设\(a=5\)\(b=110101\),我们有\(a*b=a*(100000*1+10000*1+1000*0+100*1+10*0+1*1)\),如下表所示可以换成加法运算并在加法中取模:

alt

图示:

alt

Java实现代码:

import java.util.*;
public class Solution {private long mod = 998244353;//快速乘法private long fast(long x, long y){ long res = 0;x %= mod;y %= mod;while(y != 0){if((y & 1L) != 0){//加法代替乘法,防止越界res += x; if(res >= mod)res -= mod;}y = y >> 1;x = x << 1;if(x >= mod)x -= mod;}return res;}//快速幂long Pow(long x, long y){ long res = 1;while(y != 0){//可以再往上乘一个if((y & 1L) != 0) res = fast(res, x);//叠加x = fast(x, x); //减少乘次数y = y >> 1; }return res;}public long cutRope (long number) {//不超过3直接计算if(number <= 3) return number - 1;//能整除3if(number % 3 == 0) return Pow(3, number / 3);//最后剩余1else if(number % 3 == 1) //4*3^{n-1}return fast(Pow(3, number / 3 - 1), 4); //最后剩余2else //2*3^nreturn fast(Pow(3, number / 3), 2); }
}

C++实现代码:

class Solution {
public:long long mod = 998244353;//快速乘法long long fast(long long x, long long y){ long long res = 0;x %= mod;y %= mod;while(y){if(y & 1){//加法代替乘法,防止越界res += x; if(res >= mod)res -= mod;}y = y >> 1;x = x << 1;if(x >= mod)x -= mod;}return res;}//快速幂long long Pow(long long x, long long y){ long long res = 1;while(y){//可以再往上乘一个if(y & 1) res = fast(res, x);//叠加x = fast(x, x); //减少乘次数y = y >> 1; }return res;}long long cutRope(long long number) {//不超过3直接计算if(number <= 3) return number - 1;//能整除3if(number % 3 == 0) return Pow(3, number / 3);//最后剩余1else if(number % 3 == 1) //4*3^{n-1}return fast(Pow(3, number / 3 - 1), 4); //最后剩余2else //2*3^nreturn fast(Pow(3, number / 3), 2); }
};

Python实现代码:

class Solution:def __init__(self):self.mod = 998244353#快速乘法def fast(self, x: int, y: int) -> int: res = 0x %= self.mody %= self.modwhile y:if y & 1:#加法代替乘法,防止越界res += xif res >= self.mod:res -= self.mody = y >> 1x = x << 1if x >= self.mod:x -= self.modreturn res#快速幂def Pow(self, x: int, y: int) -> int: res = 1while y:#可以再往上乘一个if y & 1: res = self.fast(res, x)#叠加x = self.fast(x, x) #减少乘次数y = y >> 1return resdef cutRope(self , number: int) -> int:#不超过3直接计算if number <= 3: return number - 1#能整除3if number % 3 == 0: return self.Pow(3, number // 3)#最后剩余1elif number % 3 == 1:#4*3^{n-1}return self.fast(self.Pow(3, number // 3 - 1), 4)#最后剩余2else: #2*3^nreturn self.fast(self.Pow(3, number // 3), 2)

复杂度分析:

  • 时间复杂度:\(O(log_2n)\),快速幂相当于二分法,因此复杂度为\(O(log_2n)\)
  • 空间复杂度:\(O(1)\),常数级变量,无额外辅助空间

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

牛客题解 | 判断一个链表是否为回文结构

牛客题库题解题目 题目链接 题目的主要信息:给定一个链表的头节点,判读该链表是否为回文结构 回文结构即正序遍历与逆序遍历结果都是一样的,类似123321 空链表默认为回文结构举一反三: 学习完本题的思路你可以解决如下题目: BM4.合并有序链表 BM5.合并k个已排序的链表 BM6…

牛客题解 | 判断是不是二叉搜索树

牛客题库题解题目 题目链接 题目主要信息:判断给定的一棵二叉树是否是二叉搜索树 二叉搜索树每个左子树元素小于根节点,每个右子树元素大于根节点,中序遍历为递增序举一反三: 学习完本题的思路你可以解决如下题目: BM30. 二叉搜索树与双向链表 BM37. 二叉搜索树的最近公共…

牛客题解 | 判断是不是完全二叉树

牛客题库题解题目 题目链接 题目主要信息:判断给定二叉树是否为完全二叉树 首先我们需要知道什么是完全二叉树:叶子节点只能出现在最下层和次下层,且最下层的叶子节点集中在树的左部。 需要注意的是,满二叉树肯定是完全二叉树,而完全二叉树不一定是满二叉树。举一反三: 学…

linux下安装 elasticsearch

一、基础环境 操作系统环境:Red Hat Enterprise Linux Server release 6.4 (Santiago) ES版本:elasticsearch-7.8.0-linux-x86_64.tar.gz Jdk:Java(TM) SE Runtime Environment (build 1.8.0_144-b01) 二、安装 1、上传安装包到/opt目录下 2、解压cd /opt # tar -zxvf elas…

mysql 啥样的索引能提高查询性能呢?

前言 在前面几章中,我们知道了页里面是如何存储的,页又是如何编排的。 这样我们知道了,如何定位到页,如何定位到行了,这些对我们索引的了解非常有帮助的。 知道这些后,那么我们如何利用索引查询呢? 也就是说我们如何利用这种数据结构呢? 是不是全部的查询都能通过索引去…

06 常用损失函数介绍

在前文中我们使用的损失函数都是均方误差(MSE,Mean Squared Error),本篇介绍一些其他的损失函数形式,以及他们的不同用途。 1. 回归任务常用损失函数 1.1 均方误差(MSE, Mean Squared Error) 均方误差(MSE)是回归任务中最常用的损失函数之一,用于衡量模型预测值与真实…

啦啦啦啦啦啦啦啦啦

啦啦啦 啦啦啦啦啦啦啦啦啦 ABC221G 神秘题,将坐标轴转 \(45\),然后 bitset 优化背包,记录路径把刚刚被更新的找出来,然后 _Find_next,每个点只会记一次。 AGC050a 神秘题,想到 \(\log\),然后发现一下位置 \(x\) 走十次能到的区间是 \([1024x,1024x+1023]\),区间长度够…

破解 vLLM + DeepSeek 规模化部署的“不可能三角”

通过 FC GPU 预留实例的闲置计费功能,企业用户能在充分利用 vLLM 的强大功能的同时找到成本、性能、稳定的最佳平衡点,并保持开发和运维的高效性。无论是将 FC vLLM 函数直接对外提供服务,还是深度集成到现有系统中,或是通过 CAP 还是魔搭来简化部署,都能找到满足您业务需…

条形码编码规则全解析:从黑白条纹到数字世界的转换密码

条形码的编码规则是将字符(数字、字母等)转换为特定黑白条纹或矩阵结构的标准化方法,核心目的是让机器能够快速、准确地识别和解析信息。以下是常见条形码编码规则的简介: 一维条形码编码规则 1. ​基本原理通过不同宽度的黑白条纹​(或空格)组合表示字符。 每个字符对应…

[汽车电子/车联网] CANoe

概述:CANoe CAN 全家桶区别: CANoe vs CanalyzerCANoe和CANalyzer使用方法类似(简直可以说 相同)。 都可用于simulation,区别在于CANalyzer只能模拟单个Node,而CANoe可以同时模拟多个Node。如果入门学习了CANoe,就不用入门学习CANalyzer了。安装指南 安装 CANoeDemo on …

璞华易研PLM荣登软服之家多项榜单,PLM+AI为流程行业提供产品创新引擎

近日,国内知名软件与服务评测平台软服之家发布了多个PLM(Product Lifecycle Management,产品全生命周期管理)榜单,帮助用户了解PLM领域表现卓越的软件产品和服务。在软服之家的多项榜单中,璞华易研PLM凭借其自主研发能力与行业深耕优势,在流程行业、电子信息、装备制造等…

ELK 原理介绍及实践详解

介绍了ELK(Elasticsearch, Logstash, Kibana)在大规模日志管理中的重要性,阐述了ELK解决日志分析的挑战,如日志收集、传输、存储和分析。文章详细讲解了ELK的组成部分,包括Filebeat的工作原理、Logstash的输入、过滤和输出阶段,以及Kibana的分析和可视化功能。此外,还提…