ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

【动态规划】力扣650:只有两个键的键盘

2022-04-26 12:03:01  阅读:145  来源: 互联网

标签:p2 键盘 p1 int 复杂度 力扣 650 质数 dp


最初记事本上只有一个字符 'A' 。你每次可以对这个记事本进行两种操作:
Copy All(复制全部):复制这个记事本中的所有字符(不允许仅复制部分字符)。
Paste(粘贴):粘贴 上一次 复制的字符。
给你一个数字 n ,你需要使用最少的操作次数,在记事本上输出 恰好 n 个 'A' 。返回能够打印出 n 个 'A' 的最少操作次数。
示例:

输入:3
输出:3
解释:
最初, 只有一个字符 'A'。
第 1 步, 使用 Copy All 操作。
第 2 步, 使用 Paste 操作来获得 'AA'。
第 3 步, 使用 Paste 操作来获得 'AAA'。

方法1:数学
首先写前几个函数值并观察规律:
1 2 3 4 5 6 7 8 9 10
| | | | | | | | | |
0 2 3 4 5 5 7 6 6 7

  • 结论 1 :f(2 * n) = f(n) + 2.
    这是因为我们需要先通过 f(n) 次操作得到 n,再复制粘贴(2步)得到 2 * n.
  • 结论 2 :对于质数 p,f(p) = p.
    这是因为要达到 p,只有复制 1,再粘贴 p - 1 次.
    从以上两个结论得到启发,我们考虑一个合数 n = p1 * p2 的达到方式,有三种途径可以达到 n:
    1 -> p1 * p2
    1 -> p1 -> p1 * p2
    1 -> p2 -> p1 * p2
    这些途径的步数分别为 p1 * p2, p1 + 1(复制) + (p2 - 1)(粘贴) = p1 + p2, p2 + 1(复制) + (p1 - 1)(粘贴) = p1 + p2.
    所以,f(p1 * p2) = p1 + p2.
    从而很容易推导出,若 n = p1 * p2 * ... * pi,其中 pi 为质数,则
    f(n) = p1 + p2 + ... + pi.
    进一步地,若 n = p1^r1 * p2^r2 * ... * pi^ri,其中 pi 为质数,ri >= 1,则
    f(n) = p1 * r1 + p2 * r2 + ... + pi * ri.
    也就是对 n 做质因数分解,并将所有质因数乘以它的幂次并相加即可
class Solution:
    def minSteps(self, n: int) -> int:
        '''
        if n = 1:
            return 0
        '''
        res = 0 # n = 1 时返回 0,规律从 n = 2 开始
        m = n
        for i in range(2, n + 1):
            while m % i == 0:
                res += i
                m //= i
                # 可以加一个判断:当n为合数,m = 1时说明已经分解完成,不需要继续。但是由于若n为质数,i要循环到n才能停止,因此i的范围为[2, n],而不能提前。(不加这一句不影响)
                if m == 1:
                    break
        return res

时间复杂度:O(n)。
空间复杂度:O(1)。

由于n为合数时i不需要遍历完,可以在这方面优化时间复杂度,但思路较复杂。

class Solution:
    def minSteps(self, n: int) -> int:
        ans = 0
        # 若n为合数,进入循环结束时n已经改变,是一个质数,进入质数阶段
        i = 2
        while i * i <= n:
            while n % i == 0:
                ans += i
                n //= i
            i += 1
        # 若n为质数,操作次数就是n,直接返回n。但若此时的n是循环结束来的,则结果为ans+n
        if n > 1:
            ans += n
        return ans

时间复杂度:O(sqrt(n)),即为质因数分解的时间复杂度。
空间复杂度:O(1)。

方法2:动态规划
也可以用动态规划根据上面的数学思路实现。
不同于以往通过加减实现的动态规划,这里需要乘除法来计算位置,因为粘贴操作是倍数增加的。我们使用一个一维数组 dp,其中位置 i 表示延展到长度 i 的最少操作次数。对于每个位置j,如果 j 可以被 i 整除,那么长度 i 就可以由长度 j 操作得到,其操作次数等价于把一个长度为 1的 A 延展到长度为 i/j。因此可以得到递推公式 dp[i] = dp[j] + dp[i/j]。

class Solution:
    def minSteps(self, n: int) -> int:
        dp = [0] * (n + 1)
        m = int(sqrt(n))
        for i in range(2, n + 1):
            dp[i] = i # 因为结果取min,初始的dp[i]就要尽量大,根据枚举发现dp[n]最大为n,所以可以初始化为而不是`inf`
            for j in range(2, m + 1):
                if i % j == 0:
                    dp[i] = dp[j] + dp[i//j]
                    break
        return dp[n]

时间复杂度:O(n×sqrt(n)),即为质因数分解的时间复杂度。
空间复杂度:O(n)。

标签:p2,键盘,p1,int,复杂度,力扣,650,质数,dp
来源: https://www.cnblogs.com/Jojo-L/p/16194186.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有