[Leetcode] 650. 2 Keys Keyboard 解题报告

探讨了在记事本上通过复制和粘贴操作生成指定数量字符的算法。介绍了三种解决方案,包括O(n^2)和O(n)的动态规划方法及O(logn)的递推法。

题目

Initially on a notepad only one character 'A' is present. You can perform two operations on this notepad for each step:

  1. Copy All: You can copy all the characters present on the notepad (partial copy is not allowed).
  2. Paste: You can paste the characters which are copied last time.

Given a number n. You have to get exactly n 'A' on the notepad by performing the minimum number of steps permitted. Output the minimum number of steps to get n 'A'.

Example 1:

Input: 3
Output: 3
Explanation:
Intitally, we have one character 'A'.
In step 1, we use Copy All operation.
In step 2, we use Paste operation to get 'AA'.
In step 3, we use Paste operation to get 'AAA'.

Note:

  1. The n will be in the range [1, 1000].

思路

1、O(n^2)的动态规划:

我们定义dp[i][j]表示在粘贴板上有j个A,文本中有i个A的情况下最小的生成步数,则递推公式为:当j < i 时,dp[i][j] = min(dp[i - j][j] + 1);而dp[i][i] = min(dp[i][j] + 1), 其中j < i。这样最后返回dp[n].begin()和dp[n].end()中间的最小数即可。不过这种O(n^2)的空间复杂度没有通过测试。

2、O(n)的动态规划:

其实可以把DP的时间复杂度降低到O(n)的。我们定义dp[i]表示生成i个A所需要的最小步数,则首先把dp[i]初始化为i(这是因为当n >= 2的时候,我们可以通过先复制一个A,然后不断粘贴实现)。然后让j从i - 1不断循环到2,检查j,如果j是i的因数,那么就可以通过首先生成j个A,然后将这j个A复制后,粘贴 (i / j) - 1次实现。这样就可以顺利通过测试了。注意到这里dp[i]只依赖于另外一个dp[j],所以这种动态规划可以进一步简化成为递归实现(见下面代码片段中的version 2)。

3、递推法:

在网上看到的既不用动态规划,也不用递归的版本,我觉得这是最优秀的一个解法了:当n >= 2的时候,我们的最优策略就是尽可能地生成n的最大因数个A(例如 n / d个),然后将其复制1遍,再粘贴 d - 1次。为了让n / d尽可能的大,我们就让d尽可能的小,所以我们让d从2开始循环。当找到一个d之后,我们接下来只需要解决生成n /d个A的问题,所以在循环中让n变为n / d即可。这样算法的时间复杂度就直接降低到了O(logn),可谓已经到了极致了。

代码

1、O(n^2)的动态规划:

class Solution {
public:
    int minSteps(int n) {
        vector<vector<int>> dp(n + 1, vector<int>(n + 1, INT_MAX));
        dp[1][0] = 0, dp[1][1] = 1;
        for (int i = 1; i <= n; ++i) {
            for (int j = 1; j < i; ++j) {
                if (dp[i - j][j] < INT_MAX) {
                    dp[i][j] = min(dp[i][j], dp[i - j][j] + 1);
                }
                if (dp[i][j] < INT_MAX) {
                    dp[i][i] = min(dp[i][i], dp[i][j] + 1);
                }
            }
        }
        return *(min_element(dp[n].begin(), dp[n].end()));
    }
};

2、O(n)的动态规划:

class Solution {
public:
    int minSteps(int n) {
        // version 1: DP
        if (n == 1) {
            return 0;
        }
        vector<int> dp(n + 1, INT_MAX);
        for (int i = 2; i <= n; ++i) {
            dp[i] = i;
            for (int j = i - 1; j > 1; --j) {
                if (i % j == 0) {
                    dp[i] = dp[j] + (i / j);
                    break;
                }
            }
        }
        return dp[n];
        // version 2: recursion
        /*if (n == 1) {
            return 0;
        }
        for (int i = n - 1; i > 1; --i) {
            if (n % i == 0) {
                return minSteps(i) + n / i;
            }
        }
        return n;*/
    }
};

3、递推法:

class Solution {
public:
    int minSteps(int n) {
        int s = 0;
        for (int d = 2; d <= n; ++d) {  // we want to make d copies of n / d 'A's
            while (n % d == 0) {
                s += d;
                n /= d;
            }
        }
        return s;
    }
};

MATLAB代码实现了一个基于多种智能优化算法优化RBF神经网络的回归预测模型,其核心是通过智能优化算法自动寻找最优的RBF扩展参数(spread),以提升预测精度。 1.主要功能 多算法优化RBF网络:使用多种智能优化算法优化RBF神经网络的核心参数spread。 回归预测:对输入特征进行回归预测,适用于连续值输出问题。 性能对比:对比不同优化算法在训练集和测试集上的预测性能,绘制适应度曲线、预测对比图、误差指标柱状图等。 2.算法步骤 数据准备:导入数据,随机打乱,划分训练集和测试集(默认7:3)。 数据归一化:使用mapminmax将输入和输出归一化到[0,1]区间。 标准RBF建模:使用固定spread=100建立基准RBF模型。 智能优化循环: 调用优化算法(从指定文件夹中读取算法文件)优化spread参数。 使用优化后的spread重新训练RBF网络。 评估预测结果,保存性能指标。 结果可视化: 绘制适应度曲线、训练集/测试集预测对比图。 绘制误差指标(MAE、RMSE、MAPE、MBE)柱状图。 十种智能优化算法分别是: GWO:灰狼算法 HBA:蜜獾算法 IAO:改进天鹰优化算法,改进①:Tent混沌映射种群初始化,改进②:自适应权重 MFO:飞蛾扑火算法 MPA:海洋捕食者算法 NGO:北方苍鹰算法 OOA:鱼鹰优化算法 RTH:红尾鹰算法 WOA:鲸鱼算法 ZOA:斑马算法
### 如何在 VSCode 中安装和配置 LeetCode 插件以及 Node.js 运行环境 #### 安装 LeetCode 插件 在 VSCode 的扩展市场中搜索 `leetcode`,找到官方提供的插件并点击 **Install** 按钮进行安装[^1]。如果已经安装过该插件,则无需重复操作。 #### 下载与安装 Node.js 由于 LeetCode 插件依赖于 Node.js 环境,因此需要下载并安装 Node.js。访问官方网站 https://nodejs.org/en/ 并选择适合当前系统的版本(推荐使用 LTS 版本)。按照向导完成安装流程后,需确认 Node.js 是否成功安装到系统环境中[^2]。 可以通过命令行运行以下代码来验证: ```bash node -v npm -v ``` 上述命令应返回对应的 Node.js 和 npm 的版本号。如果没有正常返回版本信息,则可能未正确配置环境变量。 #### 解决环境路径问题 即使完成了 Node.js 的安装,仍可能出现类似 “LeetCode extension needs Node.js installed in environment path” 或者 “command ‘leetcode.toggleLeetCodeCn’ not found” 的错误提示[^3]。这通常是因为 VSCode 未能识别全局的 Node.js 路径或者本地安装的 nvm 默认版本未被正确加载[^4]。 解决方法如下: 1. 手动指定 Node.js 可执行文件的位置 在 VSCode 设置界面中输入关键词 `leetcode`,定位至选项 **Node Path**,将其值设为实际的 Node.js 安装目录下的 `node.exe` 文件位置。例如:`C:\Program Files\nodejs\node.exe`。 2. 使用 NVM 用户管理工具调整默认版本 如果通过 nvm 工具切换了不同的 Node.js 版本,请确保设置了默认使用的版本号。可通过以下指令实现: ```bash nvm alias default <version> ``` 重新启动 VSCode 后测试功能键是否恢复正常工作状态。 --- #### 配置常用刷题语言 最后一步是在 VSCode 设置面板中的 LeetCode 插件部分定义个人习惯采用的主要编程语言作为默认提交方式之一。这样可以减少频繁修改编码风格的时间成本。 --- ### 总结 综上所述,要在 VSCode 上顺利启用 LeetCode 插件及其关联服务,除了基本插件本身外还需额外准备支持性的后台框架——即 Node.js 应用程序引擎;同时针对特定场景下产生的兼容性障碍采取针对性措施加以修正即可达成目标[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值