牛客周赛round123 G小红出千[补题][滑动窗口]

G-小红出千_牛客周赛 Round 123

根据题目 我们要选择一个长度为n的区间 令这个区间内的不重复数字最多 这一过程可以使用滑动窗口实现

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

int main() {
    ios::sync_with_stdio(0);
    cin.tie(0);
    cout.tie(0);
    int n;
    cin>>n;
    vector<int>a(n);
    for(int i=0;i<n;i++)cin>>a[i];
    auto b=a;
    sort(a.begin(),a.end());
    a.erase(unique(a.begin(),a.end()),a.end());
    int m=a.size();
    int cl=0,cr=0;
    for(int r=0,l=0;r<m;r++){  //滑动窗口
        while(l<r&&a[l]<=a[r]-n){   //窗口左端收缩  维护长度
            ++l;
        }
        if(cr-cl<r-l){      //更新最优窗口长度
            cr=r;cl=l;
        }
    }//连续最优子区间的最小值(最终顺子的起始值);
    int minn=a[cl];
    vector<int>vis(n);
    for(int i=cl;i<=cr;i++)vis[a[i]-minn]=1;    //标记原始数字
    vector<pair<int, int>> ans;  //保存修改方案
    for(int i=0,j=0;i<n;i++){//i 遍历数组中的每一个元素 j遍历vis
        //j向右移动 直到vis[i]!=1 也就是第一个需要填充的位置
        while(j<n&&vis[j]){
            ++j;
        }
        //判断当前原始牌b[i]是否存在最优子区间,且对应位置未被占用
        if(b[i]>=minn&&b[i]<minn+n&&vis[b[i]-minn]==1){//已经存在就不变  如果有重复第一次会变为2 后面会利用
            vis[b[i]-minn]=2;
        }else {
            ans.emplace_back(i+1,j+minn); //对于非有效 直接改变
            vis[j]=2;
        }
    }
    cout<<ans.size()<<'\n';
    for(auto [i,x]:ans){
        cout<<i<<' '<<x<<'\n';
    }
    return 0;
}
 
相关推荐
_深海凉_1 小时前
LeetCode热题100-寻找两个正序数组的中位数
算法·leetcode·职场和发展
旖-旎2 小时前
深搜练习(电话号码字母组合)(3)
c++·算法·力扣·深度优先遍历
谭欣辰2 小时前
C++快速幂完整实战讲解
算法·决策树·机器学习
Mr_pyx2 小时前
【LeetHOT100】随机链表的复制——Java多解法详解
算法·深度优先
AIFarmer2 小时前
【无标题】
开发语言·c++·算法
AGV算法笔记3 小时前
CVPR 2025 最新感知算法解读:GaussianLSS 如何用 Gaussian Splatting 重构 BEV 表示?
算法·重构·自动驾驶·3d视觉·感知算法·多视角视觉
勤劳的进取家4 小时前
数据链路层基础
网络·学习·算法
Advancer-4 小时前
第二次蓝桥杯总结(上)
java·算法·职场和发展·蓝桥杯
ん贤4 小时前
加密算法(对称、非对称、哈希、签名...)
算法·哈希算法
superior tigre5 小时前
78 子集
算法·leetcode·深度优先·回溯