codeforces 1418 G. Three Occurrences 区间非法状态维护 + 线段树维护区间最小值和最小值的个数

这类找区间个数题,一般可以枚举一个端点,找符合条件的另一个端点的个数。

这题我们枚举右端点,对每个数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
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值