龗孖 龗孖
首页
  • JAVA
  • 设计模式
  • 前端文章

    • JavaScript
  • 学习笔记

    • 《JavaScript教程》
    • 《JavaScript高级程序设计》
    • 《ES6 教程》
    • 《Vue》
    • 《React》
    • 《TypeScript 从零实现 axios》
    • 《Git》
    • TypeScript
    • JS设计模式总结
  • 页面

    • HTML
    • CSS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
关于
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)

靇孖

某微型企业非牛逼技术专家。
首页
  • JAVA
  • 设计模式
  • 前端文章

    • JavaScript
  • 学习笔记

    • 《JavaScript教程》
    • 《JavaScript高级程序设计》
    • 《ES6 教程》
    • 《Vue》
    • 《React》
    • 《TypeScript 从零实现 axios》
    • 《Git》
    • TypeScript
    • JS设计模式总结
  • 页面

    • HTML
    • CSS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
关于
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • JAVA

  • MQ

  • 工具

  • 微服务

  • 数据库

  • 其他

  • 程序设计

  • 算法

    • 限流算法
    • 数组中重复的数字
    • 二维数组中的查找
    • 替换空格
    • 从尾到头打印链表
    • 重建二叉树
    • 二叉树的下一个结点
    • 用两个栈实现队列
    • 4 变态跳台阶
    • 旋转数组的最小数字
    • 矩阵中的路径
    • 机器人的运动范围
    • 剪绳子
      • 题目链接
      • 题目描述
      • 解题思路
        • 贪心
        • 动态规划
    • 二进制中 1 的个数
    • 数值的整数次方
    • 打印从 1 到最大的 n 位数
    • 2 删除链表中重复的结点
    • 正则表达式匹配
    • 表示数值的字符串
    • 调整数组顺序使奇数位于偶数前面
    • 链表中倒数第 K 个结点
    • 链表中环的入口结点
    • 反转链表
    • 合并两个排序的链表
    • 树的子结构
    • 二叉树的镜像
    • 对称的二叉树
    • 顺时针打印矩阵
    • 包含 min 函数的栈
    • 栈的压入、弹出序列
    • 3 按之字形顺序打印二叉树
    • 二叉搜索树的后序遍历序列
    • 二叉树中和为某一值的路径
    • 复杂链表的复制
    • 二叉搜索树与双向链表
    • 序列化二叉树
    • 字符串的排列
    • 数组中出现次数超过一半的数字
    • 最小的 K 个数
    • 2 字符流中第一个不重复的字符
    • 连续子数组的最大和
    • 从 1 到 n 整数中 1 出现的次数
    • 数字序列中的某一位数字
    • 把数组排成最小的数
    • 把数字翻译成字符串
    • 礼物的最大价值
    • 最长不含重复字符的子字符串
    • 丑数
    • 第一个只出现一次的字符位置
    • 数组中的逆序对
    • 两个链表的第一个公共结点
    • 数字在排序数组中出现的次数
    • 二叉查找树的第 K 个结点
    • 2 平衡二叉树
    • 数组中只出现一次的数字
    • 2 和为 S 的连续正数序列
    • 2 左旋转字符串
    • 滑动窗口的最大值
    • n 个骰子的点数
    • 扑克牌顺子
    • 圆圈中最后剩下的数
    • 股票的最大利润
    • 求 1+2+3+...+n
    • 不用加减乘除做加法
    • 构建乘积数组
    • 把字符串转换成整数
    • 树中两个节点的最低公共祖先
  • 服务端
  • 算法
龗孖
2024-08-31
目录

剪绳子

# 14. 剪绳子

# 题目链接

牛客网 (opens new window)

# 题目描述

把一根绳子剪成多段,并且使得每段的长度乘积最大。

n = 2
return 1 (2 = 1 + 1)

n = 10
return 36 (10 = 3 + 3 + 4)
1
2
3
4
5

# 解题思路

# 贪心

尽可能得多剪长度为 3 的绳子,并且不允许有长度为 1 的绳子出现。如果出现了,就从已经切好长度为 3 的绳子中拿出一段与长度为 1 的绳子重新组合,把它们切成两段长度为 2 的绳子。以下为证明过程。

将绳子拆成 1 和 n-1,则 1(n-1)-n=-1<0,即拆开后的乘积一定更小,所以不能出现长度为 1 的绳子。

将绳子拆成 2 和 n-2,则 2(n-2)-n = n-4,在 n>=4 时这样拆开能得到的乘积会比不拆更大。

将绳子拆成 3 和 n-3,则 3(n-3)-n = 2n-9,在 n>=5 时效果更好。

将绳子拆成 4 和 n-4,因为 4=2*2,因此效果和拆成 2 一样。

将绳子拆成 5 和 n-5,因为 5=2+3,而 5<2*3,所以不能出现 5 的绳子,而是尽可能拆成 2 和 3。

将绳子拆成 6 和 n-6,因为 6=3+3,而 6<3*3,所以不能出现 6 的绳子,而是拆成 3 和 3。这里 6 同样可以拆成 6=2+2+2,但是 3(n - 3) - 2(n - 2) = n - 5 >= 0,在 n>=5 的情况下将绳子拆成 3 比拆成 2 效果更好。

继续拆成更大的绳子可以发现都比拆成 2 和 3 的效果更差,因此我们只考虑将绳子拆成 2 和 3,并且优先拆成 3,当拆到绳子长度 n 等于 4 时,也就是出现 3+1,此时只能拆成 2+2。

public int cutRope(int n) {
    if (n < 2)
        return 0;
    if (n == 2)
        return 1;
    if (n == 3)
        return 2;
    int timesOf3 = n / 3;
    if (n - timesOf3 * 3 == 1)
        timesOf3--;
    int timesOf2 = (n - timesOf3 * 3) / 2;
    return (int) (Math.pow(3, timesOf3)) * (int) (Math.pow(2, timesOf2));
}
1
2
3
4
5
6
7
8
9
10
11
12
13

# 动态规划

public int cutRope(int n) {
    int[] dp = new int[n + 1];
    dp[1] = 1;
    for (int i = 2; i <= n; i++)
        for (int j = 1; j < i; j++)
            dp[i] = Math.max(dp[i], Math.max(j * (i - j), dp[j] * (i - j)));
    return dp[n];
}
1
2
3
4
5
6
7
8
上次更新: 2024/11/03, 21:06:16
机器人的运动范围
二进制中 1 的个数

← 机器人的运动范围 二进制中 1 的个数→

最近更新
01
树中两个节点的最低公共祖先
10-17
02
hexo多平台多博客网站同步
09-04
03
最长不含重复字符的子字符串
09-03
更多文章>
Theme by Vdoing | Copyright © 2015-2024 Ling ma | 996.icu | 京ICP备16011424号-1
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式