题解 DIFERENC - DIFERENCIJA
题目描述
给出一个长度为
即定义一个子序列的权值为序列内最大值与最小值的差。求出所有连续子序列的权值和。
具体思路
暴力思路很好想,就是按照式子来打,然后区间最大值和区间最小值可以用数组预处理一下。
时间复杂度:
那么我们发现,如果我们的两个求和不拆掉那么时间复杂度就不正确,那么思路就很显然:计算每个
可以考虑将
如图所示,我们来考虑上图中红色部分作为最大值被计算了多少次。
-
当
,当 时 被计算了一次。 -
当
,当 时 被计算了一次。
那我们发现,
设
如果我们暴力枚举每一个点,然后暴力找上一个比它大的位置以及下一个比它大的位置,时间复杂度:
这不没优化吗?
上面这个图,显然就很有单调栈的味道。
从左往右依次枚举每一个点,维护一个单调下降的栈,如果当前的点比栈顶要大,那么更新栈顶的信息,同时将栈顶踢出栈。
然后我们就愉快的解决了这个问题。
维护最小值也是同样的道理。
时间复杂度:
Code
#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=3e5+5;
const int inf=0x3f3f3f3f;
int top,sta[N],l[N],r[N],a[N];
signed main(){
int n;scanf("%lld",&n);
for(int i=1;i<=n;i++){
scanf("%lld",&a[i]);
}
top=1;sta[top]=1;
l[1]=1;
for(int i=2;i<=n;i++){
while(top&&a[i]>a[sta[top]]){
r[sta[top]]=i-sta[top];
top--;
}
l[i]=i-sta[top];
sta[++top]=i;
}
a[n+1]=inf;
while(top&&a[n+1]>a[sta[top]]){
r[sta[top]]=n+1-sta[top];
top--;
}
int maxn=0;
for(int i=1;i<=n;i++){
maxn=maxn+l[i]*r[i]*a[i];
}
top=1;sta[top]=1;
l[1]=1;
for(int i=2;i<=n;i++){
while(top&&a[i]<a[sta[top]]){
r[sta[top]]=i-sta[top];
top--;
}
l[i]=i-sta[top];
sta[++top]=i;
}
a[n+1]=0;
while(top&&a[n+1]<a[sta[top]]){
r[sta[top]]=n+1-sta[top];
top--;
}
int minn=0;
for(int i=1;i<=n;i++){
minn=minn+l[i]*r[i]*a[i];
}
printf("%lld",maxn-minn);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 如何调用 DeepSeek 的自然语言处理 API 接口并集成到在线客服系统
· 【译】Visual Studio 中新的强大生产力特性
· 2025年我用 Compose 写了一个 Todo App