CF1359D Yet Another Yet Another Task

这题难度评得是不是太低了 qwq,它在 CF 上的过题人数甚至不到两千。

分析

我们记读入的数组为 w[]

我的思路是从左到右枚举位置 \(i\),然后找 \(i\) 最左边的点 \(x\) 使得对于 \(j\in [x, i-1]\) 有 \(w[i] \leq w[i]\),类似地找到 \(i\) 最右边的点 \(y\) 使得 \(j\in [i+1, y]\) 有 \(w[y]\leq w[i]\)。

因为最值是具有单调性的,我们可以用二分和 ST 表来实现上面的查找过程。

接下来我们从 \([x, i-1]\) 找到最大的后缀和以及从 \([i+1, y]\) 找到最大的前缀和,这一步也可以用 ST 表来解决,最后更新答案即可。

细节见代码:

#pragma GCC optimize("O3")
#include<bits/stdc++.h>
using namespace std;

#define endl '\n'
#define debug(x) cerr << #x << ": " << x << endl
#define pb push_back
#define eb emplace_back
#define set0(a) memset(a,0,sizeof(a))
#define rep(i,a,b) for(int i=(a);i<=(b);i++)
#define dwn(i,a,b) for(int i=(a);i>=(b);i--)

#define INF 0x3f3f3f3f

inline void read(int &x) {
    int s=0;x=1;
    char ch=getchar();
    while(ch<'0'||ch>'9') {if(ch=='-')x=-1;ch=getchar();}
    while(ch>='0'&&ch<='9') s=(s<<3)+(s<<1)+ch-'0',ch=getchar();
    x*=s;
}

const int N=1e5+5;

int n;
int w[N];
int Log[N];

void init(){
	Log[1]=0;
	rep(i,2,N-1) Log[i]=Log[i/2]+1;
}

struct ST{
	int f[N][25];
	
	void init(int *a){
		rep(i,1,n) f[i][0]=a[i];
	}
	
	void build(){
		rep(j,1,20) for(int i=1; i+(1<<j)-1<=n; i++)
			f[i][j]=max(f[i][j-1], f[i+(1<<j-1)][j-1]);
	}
	
	int query(int l, int r){
		int k=Log[r-l+1];
		return max(f[l][k], f[r-(1<<k)+1][k]);
	}
}a, p, q;

int pre[N], suf[N];

int find(int l, int r, int v, bool op){
	int L=l, R=r;
	if(op){
		while(L<R){
			int mid=L+R>>1;
			if(a.query(mid, r)<=v) R=mid;
			else L=mid+1;
		}
		if(a.query(L, r)>v) return 0;
		else return L;
	}
	else{
		while(L<R){
			int mid=L+R+1>>1;
			if(a.query(l, mid)<=v) L=mid;
			else R=mid-1;
		}
		if(a.query(l, L)>v) return 0;
		else return L;
	}
}

int main(){
	init();
	read(n);
	rep(i,1,n) read(w[i]), pre[i]=pre[i-1]+w[i];
	dwn(i,n,1) suf[i]=suf[i+1]+w[i];
	
	a.init(w);
	p.init(pre);
	q.init(suf);
	
	a.build(), p.build(), q.build();
	
	int res=0;
	rep(i,1,n){
		int x, y;
		x=y=0;
		
		if(i-1>0) x=find(1, i-1, w[i], 1);
		if(i+1<=n) y=find(i+1, n, w[i], 0);
		
		int t=0;
		if(x) t+=max(0, q.query(x, i-1)-suf[i]); // 这里需要取 max(0, ),因为我们有权利不取这部分。
		if(y) t+=max(0, p.query(i+1, y)-pre[i]);
		res=max(res, t);
	}
	
	cout<<res<<endl;
	
	return 0;
}

附送 debug 数据:

in:
202
28 17 17 -25 -26 -17 18 1 16 -26 -30 8 -22 17 -20 16 -23 16 -25 3 21 27 -21 -16 -8 21 -22 19 10 -11 -18 -27 -15 -4 6 -22 -30 -25 -18 -6 -27 18 -17 -7 -30 -16 -30 -17 -23 21 -16 27 -14 -18 18 -7 -4 -28 -19 -23 27 0 -8 -6 4 -3 -3 28 -30 11 -10 -2 6 -21 1 -13 -22 -22 22 -9 -23 -6 19 4 -10 -10 -12 10 -25 -3 -20 -29 28 10 7 11 17 27 -30 -20 -12 -24 16 -23 -10 -7 -29 -12 -25 -30 -18 24 11 -23 4 8 0 -11 -3 21 -2 3 -25 -18 2 -26 4 5 27 15 -11 -23 4 -8 -14 -6 13 -20 -3 1 24 4 -4 -19 -29 -11 -25 -7 -7 26 -11 23 -19 -2 -15 0 5 -3 -23 24 2 19 -14 28 -9 -26 -9 -5 -23 -7 -22 6 -6 -23 -9 8 17 -29 -12 -4 4 14 8 -24 -1 -5 -28 -23 24 17 -25 12 -29 14 13 17 -11 19 27 6 -17 3 

out:
72

=======================

in:
7
-5 7 -10 1 1 -8 3 

out:
1
上一篇:试玩RocketMQ, 事务消息, 以及NOT_CONSUME_YET消息不能被消费等问题


下一篇:矩阵问题学习笔记