zoukankan      html  css  js  c++  java
  • CF1359D Yet Another Yet Another Task

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

    分析

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

    我的思路是从左到右枚举位置 (i),然后找 (i) 最左边的点 (x) 使得对于 (jin [x, i-1])(w[i] leq w[i]),类似地找到 (i) 最右边的点 (y) 使得 (jin [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 '
    '
    #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
    
  • 相关阅读:
    C#Type类中的IsAssignableFrom、IsInstanceOfType、IsSubclassOf
    C# IsAssignableFrom & IsInstanceOfType & IsSubclassOf & Is
    VS2017桌面应用程序打包成.msi或者.exe
    C# 10个常用特性
    ADO.NET 的六大对象及其关系图
    Expression表达式树(C#)
    表达式树 Expression Trees
    特性 Attribute
    C# 特性(attribute)
    Vue中使用axios
  • 原文地址:https://www.cnblogs.com/Tenshi/p/15368852.html
Copyright © 2011-2022 走看看