/*
* 面试题14:剪绳子
* 题目:给你一根长度为n的绳子,请把绳子剪成m段(m和n都是整数,n>1并且m>1)每段绳子的长度记为k[0],k[1],...,k[m].
* 请问k[0]*k[1]*...*k[m]可能的最大乘积是多少?
* 例如,当绳子的长度为8时,我们把它剪成长度分别为2,3,3的三段,此时得到的最大乘积是18.
*/
/**
* 常规的需要O(n2)的时间复杂度和O(n)的空间复杂度的动态规划思路
* 题目的意思是:绳子至少是2,并且至少剪一刀。
*/
public int maxAfterCutting(int n)
{
if(n<2)
{
return 0;
}
if (n==2) {
return 2;
}
if (n==3) {
return 2;
}
int[] products=new int[n+1];
products[1]=1;
products[2]=2;
products[3]=3;
for(int i=4;i<=n;i++)
{
int max=0;
for(int j=1;j<=i/2;j++)
{
int product=products[j]*products[i-j];
if (max<product) {
max=product;
}
}
products[i]=max;
}
return products[n];
}
i从4开始的原因是4以及4之后的长度都有多种可能,需要使用前面的值计算。3也有两种可能,但是3不能使用循环计算的原因是,使用这个循环得到的结果是正确值,但是3后面的全部都会错,这个products数组书上已经说了是把长度为i的绳子剪成若干段(若干>=1)之后各段长度乘积的最大值。products[3]=3原因就是后面计算的时候用到f(3),那说明这个就可以等于3,已经切过了,但是如果是函数一开始长度就是3,根据题目要求,一刀不切是不行的。
简单说,就是直接算f(3)不等于3,但是f(4)里的f(3)可以等于3
而且f[0]是用不到的,可以不初始化,不初始化也是0