最初在一个记事本上只有一个字符 'A'。你每次能够对这个记事本进行两种操做:
1.Copy All (复制所有) : 你能够复制这个记事本中的全部字符(部分的复制是不容许的)。
2.Paste (粘贴) : 你能够粘贴你上一次复制的字符。
给定一个数字 n 。你须要使用最少的操做次数,在记事本中打印出刚好 n 个 'A'。输出可以打印出 n 个 'A' 的最少操做次数。
示例 1:java
输入: 3
输出: 3
解释:
最初, 咱们只有一个字符 'A'。
第 1 步, 咱们使用 Copy All 操做。
第 2 步, 咱们使用 Paste 操做来得到 'AA'。
第 3 步, 咱们使用 Paste 操做来得到 'AAA'。code
说明:
1.n 的取值范围是 [1, 1000] 。递归
若是每次Paste 'A',显然操做次数就是n,那么怎么减小次数呢?这里分为两种状况:
1.n为质数,不管剪切板里有多少个‘A'(除了1个'A'的状况),最终都没法获得n个'A',因此最少操做次数为n;
2.n为合数,获得最少操做次数的状况发生在已经有(n / i)个'A',其中i为最小的能整除n的数(i大于1),说明只要操做i次就能获得n个'A'。leetcode
class Solution { public int minSteps(int n) { if(n == 1) return 0; int i = 2; while(n % i != 0) i++; if(i == n) return n; return i + minSteps(n / i); } }
class Solution { public int minSteps(int n) { int res = 0; for (int i = 2; i <= n; i++) { while (n % i == 0) { res += i; n /= i; } } return res; } }
class Solution { public int minSteps(int n) { int[] dp = new int[n + 1]; for(int i = 2; i <= n; i++){ dp[i] = i; for(int j = 2; j < i; j++){ if(i % j == 0){ dp[i] = Math.min(dp[i], dp[i / j] + j); } } } return dp[n]; } }