[待字闺中]最大乘积

今天待字闺中(微信公共账号)发布了一道这样的题。(貌似去年也发过)

 一根绳子,长度为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);
}




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值