今天待字闺中(微信公共账号)发布了一道这样的题。(貌似去年也发过)
一根绳子,长度为n米,将其切成几段,每一段的长度都是整数。请给出一种切法,使得切成的各段绳子之间的乘积最大。(注:最少要切一下!)
我第一眼看过去也想到了动态规划,即将绳子看作以1为单位的有刻度的绳子,每次划分子问题的时候即为考虑第一刀切的位置k。
状态转移方程为:dp[i] = k * max(i - k, dp[i-k]),其中0<k<i为整数,代码如下所示:
</pre><pre name="code" class="cpp">
long long maxProb(int n)
{
vector<long long> dp;
dp.assign(n+1, 0);
dp[1] = 1;
for(int i = 2; i <= n; ++i)
{
for(int k = 1; k < i; ++k)
{
long long tmp = i-k;
if(dp[i-k] > tmp) tmp = dp[i-k];
tmp *= k;
if(tmp > dp[i]) dp[i] = tmp;
}
}
return dp[n];
}
下面还有一个非常巧妙的解法,当然不是我想出来的啦!将n写成2x+3y,得到的乘积肯定是最大的,即2^x+3^y,代码如下:
long long maxProd(int n)
{
if(n <= 1)
{
cout << "Invalid Input" << endl;
return 0;
}
if(2 == n) return 1;
else if(3 == n) return 2;
int fac3 = n / 3;
if((n - 3 * fac3) & 0x1) --fac3;
int fac2 = (n - 3 * fac3) / 2;
return (long long)(pow(2, fac2) * pow(3, fac3));
}
另外,当n不是整数,要求切断后每部分都可以不为整数时,又该怎么办呢?
首先考虑一刀将n切成两部分,分别为x和y,则有x+y=n,现在要最大化f=xy=x(n-x)=nx-x^2,当x=y=n/2时f取得最大值,也就是说将n分成两部分,为了最大化乘积我们应该在正中间将其切开。这样对于一个较大的n,我们总是可以这样一直二分下去,什么时候停止呢?当存在一小段不切开的时候比切开两部分的乘积要大时,即(n/2)^2 <= n,这时n=4。
也就是说对于给定的n值,只要他的值大于4,我们就一直将其二分下去。而对于那些初始值n就小于等于4的,因为设定了至少要切一刀,所以他们的最大乘积就是n^2/4。代码如下(不递归的话怎么写?你有什么建议呢?):
double maxProductHelper(double n)
{
if(n > 4)
{
double res = maxProductHelper(n/2);
return res * res;
} else return n;
}
double maxProduct(double n)
{
if( n <= 4) return n*n/4;
else return maxProductHelper(n);
}