JZ67 剪绳子

描述

给你一根长度为n的绳子,请把绳子剪成整数长的m段(m、n都是整数,n>1并且m>1,m<=n),每段绳子的长度记为k[1],...,k[m]。请问k[1]x...xk[m]可能的最大乘积是多少?例如,当绳子的长度是8时,我们把它剪成长度分别为2、3、3的三段,此时得到的最大乘积是18。

题解来自:https://blog.nowcoder.net/n/1174cbda3c2b46e2a15feab37b6df60b?f=comment

思路是动态规划!

因为要求最大的乘积长,如果只是剪长度为1的话,对于乘积来说无帮助,所以起码要从长度2开始减,那我们先假设减第一段,设为j(i\geq j\geq 2),i 为绳子总长度。

那么减去第一段后,还剩下 i-j 这么长。

剩下的这一段长度呢?可以剪也可以不剪,那如果要剪的话怎么最好呢?

是不是又可以把问题回溯到上一个状态dp[i-j]?等于说现在关于总长度 i 的绳子的最优解法其实就是j*dp[i-j]或者是j*(i-j)(就是要么不剪,要么就是跟减去第一段后长度的最优解法一样)

那我们就是要在这两种情况中做最大值的判断:max(dp[i], max(j * (i - j), j * dp[i - j]))

那么先算出最开始的状态:n>1,也就是dp[2]=1。(为什么不等于2?是因为必须把绳子剪成m段,m>1,所以至少是两段,长度2的绳子只能被剪成两段1)

class Solution:
    def cutRope(self, number):
        # write code here
        # dp[i]标是长度为i的绳子构成乘积的最大值
        dp = [0] * (number + 1)
        dp[2] = 1
        for i in range(3, number + 1):
            for j in range(2, i):
                # 状态转移公式
                dp[i] = max(dp[i], max(j * (i - j), j * dp[i - j]))
        return dp[number]

是剑指的最后一道题了,总用时将近一个月,感觉还是有学到很多的知识点,但是目前还不是很牢固,另外关于python的代码有一些可以通过内置函数快速解决,如果是想面试的话,后期可能还是需要看一遍C版本,思路要记得,连贯的上,同时也要能写的出来,很多代码看得懂但是不一定写的出,等复习C版本的时候要自己写!可能会再刷一遍剑指,另外再去做leecode的题目。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值