Codeforces 1167 E Range Deleting 双指针+思维

题意

给一个数列\(a​\),定义\(f(l,r)​\)为删除\(a​\)中所有满足\(l<=a_i<=r​\)的数后的数列,问有多少对\((l,r)​\),使\(f(l,r)​\)是一个非递减的数列

分析

\(f(l,r)​\)合法,则\(f(l,r+1),f(l,r+2),\dots,f(l,x)​\)也都是合法的

把每个数在\(a​\)中第一次出现的位置\(S​\)和最后一次出现的位置\(T​\),若\(f(l+1,r-1)​\)合法则满足

\(max(T_{a_1},T_{a_2},\dots,T_{a_l})<min(S_{a_r},S_{a_{r+1}},\dots,S_{a_x})\)

\(max(T_{a_1},T_{a_2},\dots,T_{a_{i-1}})<S_{a_i} ~for~all~2<=i<=l\)

\(T_{a_i}<min(S_{a_{i+1}},S_{a_{i+2}},\dots,S_{a_x}) ~for~all~r<=i<=x​\)

考虑双指针来扫\((l,r)​\),先移动\(r​\)指针使\(f(1,r-1)​\)为合法,对答案的贡献为\(x-r+2​\)

每次往后推\(l​\)\(r​\),使\(f(l+1,r-1)​\)合法

Code

#include<bits/stdc++.h>
#define fi first
#define se second
#define bug cout<<"--------------"<<endl
using namespace std;
typedef long long LL;
const double PI=acos(-1.0);
const double eps=1e-6;
const int inf=1e9;
const LL llf=1e18;
const int mod=1e9+7;
const int maxn=1e6+10;
int n,x;
int ll[maxn],rr[maxn];
int l[maxn],r[maxn];
int main(){
    ios::sync_with_stdio(false);
    //freopen("in","r",stdin);
    cin>>n>>x;
    memset(l,0x3f3f3f,sizeof(l));
    memset(ll,0x3f3f3f,sizeof(ll));
    for(int i=1,d;i<=n;i++){
        cin>>d;
        l[d]=min(i,l[d]);
        r[d]=i;
    }
    for(int i=1;i<=x;i++){
        rr[i]=max(rr[i-1],r[i]);
    }
    for(int i=x;i>=1;i--){
        ll[i]=min(l[i],ll[i+1]);
    }
    int R=x;
    LL ans=0;
    while(ll[R]>=r[R-1]&&R>=1) R--;
    for(int i=0;i<x;i++){
        if(i!=0&&l[i]<rr[i-1]) break;
        while(R<=i+1||rr[i]>ll[R]){
            R++;
        }
        ans+=x-R+2;
    }
    cout<<ans<<endl;
    return 0;
}

转载于:https://www.cnblogs.com/xyq0220/p/10875872.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值