C++单调向量算法:得到山形数组的最少删除次数

简介: C++单调向量算法:得到山形数组的最少删除次数

本题的其它解法

C++二分算法:得到山形数组的最少删除次数

题目

我们定义 arr 是 山形数组 当且仅当它满足:

arr.length >= 3

存在某个下标 i (从 0 开始) 满足 0 < i < arr.length - 1 且:

arr[0] < arr[1] < … < arr[i - 1] < arr[i]

arr[i] > arr[i + 1] > … > arr[arr.length - 1]

给你整数数组 nums ,请你返回将 nums 变成 山形状数组 的 最少 删除次数。

示例 1:

输入:nums = [1,3,1]

输出:0

解释:数组本身就是山形数组,所以我们不需要删除任何元素。

示例 2:

输入:nums = [2,1,1,5,6,2,3,1]

输出:3

解释:一种方法是将下标为 0,1 和 5 的元素删除,剩余元素为 [1,5,6,3,1] ,是山形数组。

参数范围

3 <= nums.length <= 1000

1 <= nums[i] <= 109

题目保证 nums 删除一些元素后一定能得到山形数组。

分析

本题可以转换成:最长山形数组,再进一步转换成最长升序子序列

时间复杂度

时间复杂度O(nlogn)。分两步:一,寻找左半部分。二,寻找右半部分。每步枚举每个山顶,时间复杂度O(n),每个山顶二分查找一次,时间复杂度O(logn)。

vLenToMin

vLenToMin[i]的含义是 长度为i+1 的子序列 的结尾,如果有多个符合的子序列,取结尾最小的。

比如:

原始数组 vLenToMin
1 1
1 2 {1}->{1,2}
2 1 {2}->{1}
1 2 3 {1}->{1,2
1 3 3 4 5 {1}->{1,3}->{1,3,4}->{1,3,4,5}
1 3 5 4 {1}->{1,3}->{1,3,5}->{1,3,4}

总结

一,只会在尾部增加元素。不会在其它位置增加元素。

二,不会删除元素。

三,会替换元素。

四,严格递增。

五,所有的数都小于当前值时,在末尾增加,显然是升序。

六,it第一个大于等于n的迭代器,之前的元素一定严格小于it,而it<=n,故前面元素一定小于n。

后面的元素不会ij等于it,否则它就是*it。[ii,…)都大于等于n,所有ij不会小于n。

七,规则五和六保证了vLenToMin永远严格递增。

代码

核心代码

class Solution {
public:
  int minimumMountainRemovals(vector<int>& nums) {
    vector<int> vLeftLen,vRightLen;
    Do(vLeftLen, nums);
    Do(vRightLen, vector<int>(nums.rbegin(), nums.rend()));
    std::reverse(vRightLen.begin(), vRightLen.end());
    int iMaxLen = 0;
    for (int i = 1; i+1 < nums.size(); i++)
    {
      if ((vLeftLen[i] > 1) && (vRightLen[i] > 1))
      {
        iMaxLen = max(iMaxLen, vLeftLen[i] + vRightLen[i] - 1);
      }
    }
    return nums.size() - iMaxLen;
  }
  void Do(vector<int>& vLen, const vector<int> nums)
  {
    vector<int> vLenToMin;//vLenToMin[i]的含义是 长度为i+1 的子序列 的结尾,如果有多个符合的子序列,取结尾最小的。
    for (const auto& n : nums)
    {
      auto it = std::lower_bound(vLenToMin.begin(), vLenToMin.end(), n);
      vLen.emplace_back(it - vLenToMin.begin() + 1);
      if (vLenToMin.end() == it)
      {
        vLenToMin.emplace_back(n);
      }
      else
      {
        if (n < *it)
        {
          *it = n;
        }
      }
    }
  }
};

测试用例

template
void Assert(const T& t1, const T& t2)
{
assert(t1 == t2);
}
template
void Assert(const vector& v1, const vector& v2)
{
if (v1.size() != v2.size())
{
assert(false);
return;
}
for (int i = 0; i < v1.size(); i++)
{
Assert(v1[i], v2[i]);
}
}
int main()
{
vector nums;
int res;
{
Solution slu;
nums = { 1,3,1 };
res = slu.minimumMountainRemovals(nums);
Assert(0, res);
}
{
Solution slu;
nums = { 2, 1, 1, 5, 6, 2, 3, 1 };
res = slu.minimumMountainRemovals(nums);
Assert(3, res);
}
{
Solution slu;
nums = { 9, 8, 1, 7, 6, 5, 4, 3, 2, 1 };
res = slu.minimumMountainRemovals(nums);
Assert(2, res);
}
{
Solution slu;
nums = { 100, 92, 89, 77, 74, 66, 64, 66, 64 };
res = slu.minimumMountainRemovals(nums);
Assert(6, res);
}
{
Solution slu;
nums = { 1, 2, 1, 3, 4, 4 };
res = slu.minimumMountainRemovals(nums);
Assert(3, res);
}
//CConsole::Out(res);

}

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。

https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程

https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版

https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛


相关文章
|
9天前
|
算法 C++
算法笔记:递归(c++实现)
算法笔记:递归(c++实现)
|
5天前
|
存储 算法 Go
算法学习:数组 vs 链表
算法学习:数组 vs 链表
12 0
|
2天前
|
算法 数据处理 C++
C++一分钟之-迭代器与算法
【6月更文挑战第21天】C++ STL的迭代器统一了容器元素访问,分为多种类型,如输入、输出、前向、双向和随机访问。迭代器使用时需留意失效和类型匹配。STL算法如查找、排序、复制要求特定类型的迭代器,注意容器兼容性和返回值处理。适配器和算法组合增强灵活性,但过度使用可能降低代码可读性。掌握迭代器和算法能提升编程效率和代码质量。
21 3
|
6天前
|
算法 前端开发 Linux
【常用技巧】C++ STL容器操作:6种常用场景算法
STL在Linux C++中使用的非常普遍,掌握并合适的使用各种容器至关重要!
32 10
|
4天前
|
存储 算法 安全
C++一分钟之-数组与指针基础
【6月更文挑战第19天】在C++中,数组和指针是核心概念,数组是连续内存存储相同类型的数据,而指针是存储内存地址的变量。数组名等同于指向其首元素的常量指针。常见问题包括数组越界、尝试改变固定大小数组、不正确的指针算术以及忘记释放动态内存。使用动态分配和智能指针可避免这些问题。示例代码展示了安全访问和管理内存的方法,强调了实践的重要性。
20 3
|
8天前
|
算法 C++
【数据结构与算法】:关于时间复杂度与空间复杂度的计算(C/C++篇)——含Leetcode刷题-2
【数据结构与算法】:关于时间复杂度与空间复杂度的计算(C/C++篇)——含Leetcode刷题
|
8天前
|
算法 C++
【数据结构与算法】:关于时间复杂度与空间复杂度的计算(C/C++篇)——含Leetcode刷题-1
【数据结构与算法】:关于时间复杂度与空间复杂度的计算(C/C++篇)——含Leetcode刷题
|
8天前
|
存储 算法 C++
【数据结构与算法】:带你手搓顺序表(C/C++篇)
【数据结构与算法】:带你手搓顺序表(C/C++篇)
|
9天前
|
算法
【经典LeetCode算法题目专栏分类】【第10期】排序问题、股票问题与TOP K问题:翻转对、买卖股票最佳时机、数组中第K个最大/最小元素
【经典LeetCode算法题目专栏分类】【第10期】排序问题、股票问题与TOP K问题:翻转对、买卖股票最佳时机、数组中第K个最大/最小元素
|
9天前
|
C++
C++数组中插入元素。
C++数组中插入元素。