C++单调向量算法应用:所有子数组中不平衡数字之和

涉及知识点

单调向量

题目

一个长度为 n 下标从 0 开始的整数数组 arr 的 不平衡数字 定义为,在 sarr = sorted(arr) 数组中,满足以下条件的下标数目:

0 <= i < n - 1 ,和

sarri+1 - sarri > 1

这里,sorted(arr) 表示将数组 arr 排序后得到的数组。

给你一个下标从 0 开始的整数数组 nums ,请你返回它所有 子数组 的 不平衡数字 之和。

子数组指的是一个数组中连续一段 非空 的元素序列。

示例 1:

输入:nums = 2,3,1,4

输出:3

解释:总共有 3 个子数组有非 0 不平衡数字:

  • 子数组 3, 1 ,不平衡数字为 1 。
  • 子数组 3, 1, 4 ,不平衡数字为 1 。
  • 子数组 1, 4 ,不平衡数字为 1 。
    其他所有子数组的不平衡数字都是 0 ,所以所有子数组的不平衡数字之和为 3 。
    示例 2:
    输入:nums = 1,3,3,3,5
    输出:8
    解释:总共有 7 个子数组有非 0 不平衡数字:
  • 子数组 1, 3 ,不平衡数字为 1 。
  • 子数组 1, 3, 3 ,不平衡数字为 1 。
  • 子数组 1, 3, 3, 3 ,不平衡数字为 1 。
  • 子数组 1, 3, 3, 3, 5 ,不平衡数字为 2 。
  • 子数组 3, 3, 3, 5 ,不平衡数字为 1 。
  • 子数组 3, 3, 5 ,不平衡数字为 1 。
  • 子数组 3, 5 ,不平衡数字为 1 。
    其他所有子数组的不平衡数字都是 0 ,所以所有子数组的不平衡数字之和为 8 。
    参数范围
    1 <= nums.length <= 1000
    1 <= numsi <= nums.length

分析

时间复杂度

O(n*logn),共五步,四步是是O(n),一步是O(nlogn),故总时间复杂度是O(nlogn)。

排在首位

左边没有数小于等于numsi,右边没有数据小于numsi

原理

排序时,相同的数字相同顺序不变。依次枚举nums个元素,再计算所有子数组的不平衡数之和。如果左边有数等于numsi或numsi-1,则numsi一定不是不平衡数字。处理numsi时,包括numsi的子数组l,r,l其取值范围为(-1,i],r取值范围为i,m_c)。令l1是nums\[l1numsi或numsl1+1numsi,l1取值范围为(-1,i)如果有多个l1,取最大值。令r1取值范围为(i,m_c),令numsr1==numsi-1,如果有多个r1,取最小值。如果l1不存在,取-1;如果r1不存在取m_c。如果l取0,il或r取r1,m_c),则nums\[i一定不是平衡数。

条件一 0,il或r取[r1,m_c)
条件二 排在首位
情况一 条件一成立 条件二一定不成立 一定不是不平衡数
情况二 条件一不成立 条件二成立 一定不是不平衡数
情况二 条件一不成立 条件二不成立 不平衡数

由于条件一和条件二不可能同时成立,所以情况二,可以简化为:条件二成立。

变量解释

num1 情况二和情况三
num2 情况二

单调向量

如果l11 < l12,且numsl11 >= numsl12,则l11被淘汰。这意味者:qLeft是单调递增,方便使用二分查找,

步骤

一,计算l1。

二,计算r1。

三,计算l2。

四,计算r2。

五,枚举numsi的不平衡数。

代码

class Solution {

public:

int sumImbalanceNumbers(vector& nums) {

m_c = nums.size();

const int iMaxValue = *std::max_element(nums.begin(), nums.end());

vector vLeftRange(m_c),vRightRange(m_c);

{

vector vLeft(iMaxValue + 1, -1);

for (int i = 0; i < m_c; i++)

{

vLeftRangei = max(vLeftnums\[i], vLeftnums\[i - 1]);

vLeftnums\[i] = i;

}

}

{

vector vRight(iMaxValue + 1, nums.size());

for (int i =m_c-1 ; i >= 0 ; i-- )

{

vRightRangei = vRightnums\[i-1];

vRightnums\[i] = i;

}

}

vector vLeftRange2(m_c), vRightRange2(m_c);

{

vector<pair<int, int>> qLeft;

for (int i = 0; i < m_c; i++)

{

auto it1 = std::upper_bound(qLeft.begin(), qLeft.end(), std::make_pair(numsi + 1, -1));

int left = (qLeft.begin() == it1) ? -1 : std::prev(it1)->second;

vLeftRange2i = left;

while (qLeft.size() && (qLeft.back().first >= numsi))

{

qLeft.pop_back();

}

qLeft.emplace_back(numsi, i);

}

}

{

vector<pair<int, int>> qRight;

for (int i = m_c - 1; i >= 0; i--)

{

auto it2 = std::upper_bound(qRight.begin(), qRight.end(), std::make_pair(numsi, -1));

auto right = (qRight.begin() == it2) ? m_c : std::prev(it2)->second;

vRightRange2i = right;

while (qRight.size() && (qRight.back().first >= numsi))

{

qRight.pop_back();

}

qRight.emplace_back(numsi, i);

}

}

int iRet = 0;

for (int i = 0; i < m_c; i++)

{

int num1 = (i - vLeftRangei) * (vRightRangei - i);//左边不包括numsi和numsi-1,右边不包括numsi的数量

int num2 = (i - vLeftRange2i) * (vRightRange2i - i);//左边全部都比他大,右边大于等于,也就是排到最左边

iRet += num1 - num2;

}

return iRet;

}

int m_c;

};

核心代码

测试

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(v1i == v2i);

}

}

template

void Assert(const T& t1, const T& t2)

{

assert(t1 == t2);

}

int main()

{

Solution slu;

vector nums = {2,3,1,3 };

int res;

复制代码
res = slu.sumImbalanceNumbers(nums);
Assert(3 ,res);




//CConsole::Out(res);

}

2023年8月旧代码一

class Solution {

public:

int sumImbalanceNumbers(vector& nums) {

m_c = nums.size();

int iRet = 0;

for (int i = 0; i < m_c; i++)

{

int iMin = numsi, iMax = numsi;

int iBalanceNum = 0;

std::unordered_set mVis;

mVis.emplace(numsi);

for (int j = i+1; j < m_c; j++)

{

const int& n = numsj;

if (mVis.count(n))

{

iRet += iBalanceNum;

continue;

}

if (n < iMin)

{

if (!mVis.count(n + 1))

{

iBalanceNum++;

}

iMin = n;

}

else if (n > iMax)

{

if (!mVis.count(n - 1))

{

iBalanceNum++;

}

iMax = n;

}

else

{

if (mVis.count(n - 1)&& mVis.count(n + 1))

{

iBalanceNum--;

}

if (mVis.count(n - 1) || mVis.count(n + 1))

{

}

else

{

iBalanceNum++;

}

}

iRet += iBalanceNum;

mVis.emplace(n);

}

}

return iRet;

}

int m_c;

};

2023年8月旧代码二

class Solution {

public:

int sumImbalanceNumbers(vector& nums) {

m_c = nums.size();

int mVis1001 = { 0 };

int iRet = 0;

for (int i = 0; i < m_c; i++)

{

int iMin = numsi, iMax = numsi;

int iBalanceNum = 0;

memset(mVis, 0, sizeof(mVis));

mVisnums\[i]= true;

for (int j = i+1; j < m_c; j++)

{

const int& n = numsj;

if (mVisn)

{

iRet += iBalanceNum;

continue;

}

if (n < iMin)

{

if (!mVisn + 1)

{

iBalanceNum++;

}

iMin = n;

}

else if (n > iMax)

{

if (!mVisn - 1)

{

iBalanceNum++;

}

iMax = n;

}

else

{

if (mVisn - 1 && mVisn + 1)

{

iBalanceNum--;

}

if (mVisn - 1 || mVisn + 1)

{

}

else

{

iBalanceNum++;

}

}

iRet += iBalanceNum;

mVisn=true;

}

}

return iRet;

}

int m_c;

};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步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

充满正能量得对大家说
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
墨家名称的来源:有所得以墨记之。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17

或者 操作系统:win10 开发环境:

VS2022 C++17

相关推荐
.千余12 小时前
【C++】C++继承入门(下):友元、静态成员与菱形继承的底层逻辑
开发语言·c++·笔记·学习·其他
初中就开始混世的大魔王12 小时前
6 Fast DDS-传输层
开发语言·c++·中间件·信息与通信
退休倒计时12 小时前
【每日一题】LeetCode 142. 环形链表 II TypeScript
算法·leetcode·链表·typescript
popcorn_min13 小时前
Digits 手写数字识别:随机森林多分类 + 像素级特征热力图
算法·随机森林·分类
liulilittle13 小时前
拥塞控制:排水终止的两种决策:OR 与 AND
网络·tcp/ip·计算机网络·算法·信息与通信·tcp·通信
weixin_3077791314 小时前
从脚本执行到智能体协作:AI辅助测试能力的范式重构
运维·开发语言·人工智能·算法·测试用例
量化君也14 小时前
从回测到全自动实盘交易,全天候策略需要经历哪些改造?
大数据·人工智能·python·算法·金融
代码中介商14 小时前
C++ 智能指针完全指南(三):weak_ptr 与循环引用
开发语言·c++
fox_lht14 小时前
第十五章 函数式语言:迭代器和闭包
开发语言·后端·学习·算法·rust
BestOrNothing_201514 小时前
ROS2 C++ 小车控制完整实战(二):自定义 msg 消息发布与订阅保姆级教程
c++·ros2·subscriber·publisher·msg·topic通信·自定义接口