offer 14 i/ ii剪绳子/动态规划

题目

https://leetcode-cn.com/problems/jian-sheng-zi-lcof/
在这里插入图片描述

方法一:动态规划 时间O(n^2) 空间O(n)

java

class Solution {
    public int cuttingRope(int n) {
        int[] dp = new int[n+1];
        for(int i = 2; i <= n; i++){
            for(int j = 1; j <= i-1; j++){
                dp[i] = Math.max(dp[i],Math.max(j*dp[i-j],j*(i-j)));
            }
        }
        return dp[n];
    }
}

js

var cuttingRope = function(n) {
    var dp = new Array(n+1).fill(0);
    for(var i = 2; i < n+1; i++){
        for(var j = 1; j <= i-1; j++){
            dp[i] = Math.max(dp[i],Math.max(j*dp[i-j],j*(i-j)));
        }
    }
    return dp[n];
};

方法二:贪心算法,数学推导 时间O(n) ,空间O(1)

https://leetcode-cn.com/problems/jian-sheng-zi-lcof/solution/mian-shi-ti-14-i-jian-sheng-zi-tan-xin-si-xiang-by/
核心思路是:尽可能把绳子分成长度为3的小段,这样乘积最大

1.如果 n == 2,返回1,如果 n == 3,返回2,两个可以合并成n小于4的时候返回n - 1
2.如果 n == 4,返回4
3.如果 n > 4,分成尽可能多的长度为3的小段,每次循环长度n减去3,乘积res乘以3;最后返回时乘以小于等于4的最后一小段(最后一小段如果是1,2,3,正好就是其不需划分乘起来值最大)
4.以上2和3可以合并

java

class Solution {
    public int cuttingRope(int n) {
        if(n < 4){
            return n - 1;
        }
        int res = 1;
        while(n > 4){
            res *= 3;
            n -= 3;
        }
        return res * n;
    }
}
/**
 * @param {number} n
 * @return {number}
 */
var cuttingRope = function(n) {
    if(n < 4){
        return n-1;
    }
    var res = 1;
    while(n >4){
        res *= 3;
        n -= 3;
    }
    return res * n;
};

ii题目 n的范围改变(2,1000),此时动态规划可能会运行超时。

在这里插入图片描述

贪心

步骤如下:

如果 n == 2,返回1,如果 n == 3,返回2,两个可以合并成n小于4的时候返回n - 1
如果 n == 4,返回4
如果 n > 4,分成尽可能多的长度为3的小段,每次循环长度n减去3,乘积res乘以3;最后返回时乘以小于等于4的最后一小段;每次乘法操作后记得取余就行
以上2和3可以合并

JS

/**
 * @param {number} n
 * @return {number}
 */
var cuttingRope = function(n) {
    if(n < 4) return n-1;
    var res = 1;
    while(n > 4){
        res = res*3%1000000007;
        n -= 3;
    }
    return res*n%1000000007;
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值