题解 | #剪绳子(进阶版)- 数学证明#

剪绳子(进阶版)

http://www.nowcoder.com/practice/106f666170554379ab1974e5a601e741

描述

题目描述

首先给定了我们一个长度为nn的绳子, 然后我们需要把这个绳子分成mm段, 然后我们问我们这个mm段的最大的乘积是多少

题解

解法一: 我们暴力思考, 进行一个数学推导

实现思路

我们可以对这个考虑一下数学思想的一个证明

F6EC44753BED1D5077F3C7F918BF87F4

根据算数几何均值不等式可以知道以下结论

13E5EAC28A0958BB99D81140E38F5565

得到推论: 将绳子以相等的长度分为多段得到的乘积最大

我们将绳子按照xx长度等分成为mm段, n==mxn == m * x, 然后我们的乘积就是xmx ^ m, 然后我们可以得到这样的一个式子

40A1700C1629469B7A52306D64CC54B1

我们其实就是对于最后的式子求极大值, 这个根据求导可得, 这里不过于赘述, 最后我们可以知道驻点是x==ex == e

然后我们的切分长度要求是为整数, 我们考虑2233哪一个更是接近我们的ee

我们可以把这个2233分别带入我们上述的x1xx ^ \frac{1}{x}, 我们可以发现我们的33的值更大, 所以我们每次最好的方式就是每次都尽可能的切成长度为33

那么我们的算法流程就是可以确定了下来

  1. 如果我们按照长度为33进行一个切分, 如果最后剩下的是00, 不用考虑了
  2. 如果最后留下来的是22, 我们不拆分
  3. 如果最后留下来的是11, 我们把之前的一个33变成2+22 + 2

代码实现

class Solution {
    const int mod = 998244353;
   public:
    long long cutRope(long long number) {
        if (number <= 3) return number - 1;
        // 如果不到3我们可以直接计算
        long long cnt = 0;
        while (number > 4) {
            cnt += 1;
            number -= 3;
        }
        // 计算我们的这个有多少段3
        long long res = 1;
        for (int i = 1; i <= cnt; i++) res = (res % mod * 3 % mod) % mod;
        // 得到我们最后的答案
        return res * number % mod;
    }
};

时空复杂度分析

时间复杂度: O(n)O(n)

理由如下: 我们需要运算n/3n / 3次, 是O(n)O(n)级别的

空间复杂度: O(1)O(1)

理由如下: 我们只引入了常数级别的空间

解法二: 快速幂

实现思路:

首先我们要考虑这么一个问题, 这个数据范围如果我们正常的O(n)O(n)的做法肯定是会TLETLE, 那么我们就是需要考虑一种算法这种的时间复杂度要可以达到n\sqrt{n}或者是lognlogn, 那么我们再去考虑我们上面慢在哪里, 我们是不是慢在一次次的寻找上面, 那么我们其实可以发现我们最简单的一个问题, 如果优化这个方面, 我们可以得到以下的算法思路

  1. 如果我们可以整除33, 那么我们可以直接计算3n33 ^ \frac{n}{3}
  2. 如果我们除3311, 那么我们结果就是43n314 * 3 ^ {\frac{n}{3} - 1}
  3. 如果我们余22, 我们结果就是23n32 * 3 ^ \frac{n}{3}

代码实现

class Solution {
    const long long mod = 998244353;

   public:
    long long qpow(long long a, long long b) {
        long long res = 1 % mod;
        a %= mod;
        while (b) {
            if (b & 1) res = res * a % mod;
            // 如果我们这位的二进制有值, 那么我们可以把这位乘上他的权重
            a = a * a % mod;
            // 更新权重
            b >>= 1;
            // 调整我们的b
        }
        return res;
    }
    // 快速幂
    long long cutRope(long long number) {
        if (number <= 3) return number - 1;
        // 如果不到3我们可以直接计算
        if (number % 3 == 0) return qpow(3, number / 3) % mod;
        // 如果正好整除
        else if (number % 3 == 1) return qpow(3, number / 3 - 1) * 4 % mod;
        // 如果最后余1
        else return qpow(3, number / 3) * 2 % mod;
        // 如果余是2 
    }
};

时空复杂度分析

时间复杂度: O(logn)O(logn)

理由如下: 我们的快速幂本质就是我们对我们的二进制位和它们的权值进行一个计算, 那么我们也就是需要遍历我们的所有的二进制位, 然后我们一个数的二进制位数也就是lognlogn级别的

空间复杂度: O(1)O(1)

理由如下: 我们未引用额外的空间

机试题目题解 文章被收录于专栏

主要是机试题目的题目讲解和做法

全部评论

相关推荐

10-09 22:05
666 C++
找到工作就狠狠玩CSGO:报联合国演讲,报电子烟设计与制造
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务