这类找区间个数题,一般可以枚举一个端点,找符合条件的另一个端点的个数。
这题我们枚举右端点,对每个数x而言,其作为右端点的时候,满足条件的左端点必须是其前面2个x与前面3个x之间的点。
我们用lst[i][j],当前状态下,从后往前第j个i的位置。
显然:对于每个值i,都有:在区间[lst[i][3]+1 - lst[i][2]] 合法,其余点做为左端点均不合法。
由此有了做法:
从左往右枚举右端点,对于每个值,当前状态下,某个点可以作为左端点,则其值+1,否则不变。
显然,只有满足点值为0的点才是符合条件的左端点。
我们枚举右端点的同时不断改变每个点的值,始终保持对每个值,不在[lst[i][3]+1 - lst[i][2]] 的点的值均加1.
然后求出最小值是否为0,及最小值的个数,就是满足条件的左端点的个数。
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
#define re register
#define ls (o<<1)
#define rs (o<<1|1)
#define pb push_back
const double PI= acos(-1.0);
const int M = 5e5+7;
ll tr[M<<2],nm[M<<2],tg[M<<2];
int lst[M][5],a[M];
void pu(int o)
{
tr[ls]+=tg[o];tg[ls]+=tg[o