Leetcode3196. 最大化子数组的总成本

Every day a Leetcode

题目来源:3196. 最大化子数组的总成本

解法1:记忆化搜索

因为要解决的问题都形如「a[0] 到 a[i] 的最大成本和」,所以用它作为本题的状态定义 dfs(i)。

分类讨论:

  • 分成长为 1 的子数组,即 a[i] 单独作为一个长为 1 的子数组,接下来需要解决的问题为:a[0] 到 a[i−1] 的最大成本和,即 dfs(i)=dfs(i−1)+a[i]。
  • 分成长为 2 的子数组,即 a[i−1] 和 a[i] 作为一个长为 2 的子数组,接下来需要解决的问题为:a[0] 到 a[i−2] 的最大成本和,即 dfs(i)=dfs(i−2)+a[i−1]−a[i]。

这两种情况取最大值,就得到了 dfs(i),即:dfs(i)=max(dfs(i−1)+a[i],dfs(i−2)+a[i−1]−a[i])。

递归边界:dfs(−1)=0,dfs(0)=a[0]。

递归入口:dfs(n−1),也就是答案。

代码:

#
# @lc app=leetcode.cn id=3196 lang=python3
#
# [3196] 最大化子数组的总成本
#

# @lc code=start
class Solution:
    def maximumTotalCost(self, nums: List[int]) -> int:

        @cache
        def dfs(i: int) -> int:
            if i < 0:
                return 0
            if i == 0:
                return nums[0]
            return max(dfs(i - 1) + nums[i], dfs(i - 2) + nums[i - 1] - nums[i])
        return dfs(len(nums) - 1)
# @lc code=end

结果:

在这里插入图片描述

复杂度分析:

时间复杂度:O(n),其中 n 是数组 nums 的长度。

空间复杂度:O(n),其中 n 是数组 nums 的长度。

解法2:

代码:

/*
 * @lc app=leetcode.cn id=3196 lang=cpp
 *
 * [3196] 最大化子数组的总成本
 */

// @lc code=start
class Solution
{
public:
    long long maximumTotalCost(vector<int> &nums)
    {
        int n = nums.size();
        vector<long long> dp(n + 1);
        // 初始化
        dp[0] = 0;
        dp[1] = nums[0];
        // 动态规划
        for (int i = 2; i <= n; i++)
            dp[i] = max(dp[i - 1] + nums[i - 1], dp[i - 2] + nums[i - 2] - nums[i - 1]);
        return dp[n];
    }
};
// @lc code=end

结果:

在这里插入图片描述

复杂度分析:

时间复杂度:O(n),其中 n 是数组 nums 的长度。

空间复杂度:O(n),其中 n 是数组 nums 的长度。

### 解题思路 LeetCode 第80题要求删除一个有序数组中出现次数超过两次的重复元素,使得每个元素最多出现两次,并返回去重后的数组长度。此问题可以通过双指针的方法来实现。 - 由于数组是有序的,因此相同的元素一定是连续出现的。 - 使用两个指针:一个用于遍历数组(`i`),另一个用于记录不重复部分的位置(`k`)。 - 如果当前元素 `nums[i]` 不等于 `nums[k]`,说明遇到了一个新的元素,则将其放到 `nums[k+1]` 的位置,并将 `k` 后移一位。 - 如果当前元素前一个元素相同,则跳过该元素,直到遇到不同的元素为止。 - 最终返回 `k + 1` 作为新数组的长度。 ### C语言实现 以下是针对 LeetCode 80 题的 C 语言实现代码: ```c #include <stdio.h> int removeDuplicates(int* nums, int numsSize) { if (numsSize <= 2) return numsSize; // 如果数组长度小于等于2,直接返回原长度 int k = 1; // 指向结果数组的最后一个有效位置 for (int i = 2; i < numsSize; i++) { if (nums[i] != nums[k - 1]) { // 当前元素结果数组倒数第二个元素不同 nums[++k] = nums[i]; // 将其加入结果数组 } } return k + 1; // 返回新数组长度 } ``` ### 测试代码 以下是一个简单的测试用例,用于验证上述函数是否正确工作: ```c #include <stdio.h> int main() { int nums[] = {1, 1, 1, 2, 2, 3}; int numsSize = sizeof(nums) / sizeof(nums[0]); int newLength = removeDuplicates(nums, numsSize); printf("New length: %d\n", newLength); printf("Modified array: "); for (int i = 0; i < newLength; i++) { printf("%d ", nums[i]); } printf("\n"); return 0; } ``` ### 输出示例 运行上述测试代码后,输出应为: ``` New length: 5 Modified array: 1 1 2 2 3 ``` ### 算法复杂度分析 - **时间复杂度**:O(n),其中 n 是数组的长度。只需要一次遍历即可完成操作。 - **空间复杂度**:O(1),算法在原地修改数组,没有使用额外的空间[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

UestcXiye

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值