【题目链接】click here~~

【题目大意】:

给出一个数列,问当中存在多少连续子序列,子序列的最大值-最小值<k

【思路】:枚举数列左端点。然后二分枚举右端点,用ST算法求区间最值。(或用单调队列的思路)

代码:

#include <bits/stdc++.h>
using namespace std;
const int N=1e5+10;
typedef long long LL;
#define Max(a,b) a>b?a:b
#define Min(a,b) a>b? b:a
#define mem(a,b) memset(a,b,sizeof(a))
int arr[N];
int n,k,m,tmp;
int dp_max[N][20],dp_min[N][20];
void rmq_init(){
for(int i=1; i<=n; ++i) dp_max[i][0]=dp_min[i][0]=arr[i];
double limit=log(n)/log(2.0); // 换底公式
for(int j=1; j<=(int)limit; ++j){
for(int i=1; i+(1<<j)-1<=n; ++i){
dp_max[i][j]=Max(dp_max[i][j-1],dp_max[i+(1<<(j-1))][j-1]);
dp_min[i][j]=Min(dp_min[i][j-1],dp_min[i+(1<<(j-1))][j-1]);
}
}
} int rmq_max(int L,int R){ // 查询[L,R]之间的最大值
int k=floor(log2((double)(R-L+1)));
return Max(dp_max[L][k], dp_max[R - (1<<k) + 1][k]);
} int rmq_min(int L, int R){ // 查询[L,R]之间的最小值
int k=floor(log2((double)(R-L+1)));
return Min(dp_min[L][k], dp_min[R - (1<<k) + 1][k]);
} int solve(int L,int R){
int k=floor(log2((double)(R-L+1)));
int maxx=Max(dp_max[L][k],dp_max[R - (1<<k) + 1][k]);
int minn=Min(dp_min[L][k],dp_min[R - (1<<k) + 1][k]);
return maxx-minn;
} inline LL read(){
int c=0,f=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){c=c*10+ch-'0';ch=getchar();}
return c*f;
} int main(){
int t;t=read();
while(t--){
n=read();k=read();
for(int i=1; i<=n; ++i){
arr[i]=read();
}
rmq_init();
LL ans=0;
for(int i=1; i<=n; ++i){//枚举左端点,二分右端点,ST求最值
int ll=i,rr=n;
while(ll<=rr){
int mid=(ll+rr)>>1;
if(solve(i,mid)>=k) rr=mid-1;
else ll=mid+1;
}
if(solve(i,rr)<k)ans+=(rr-i+1);
else ans+=(ll-i+1);
}
printf("%I64d\n",ans);
} return 0;
}

单调队列:

【思路】:

O(n)复杂度

用两个单调队列维护最大值,最小值,相当于双指针,初始,第一个第二个指针指向第一个数据,第一个指针按顺序不断向队尾加入数据,当最大值最小值的差大于等于k后,意味着新加入的这个不能作用于当前第二个指针的位置,也就能计算出,以第二个指针位置開始的连续子序列的个数。最后统计就能够了。

代码:

#include <bits/stdc++.h>
using namespace std;
const int N=1e5+10;
typedef long long LL;
#define Max(a,b) a>b?a:b
#define Min(a,b) a>b?b:a
#define mem(a,b) memset(a,b,sizeof(a))
int arr[N];
int i,j,n,k,m,tmp;
deque <int >deq_max,deq_min;// maxvalue minvalue
inline LL read(){
int c=0,f=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){c=c*10+ch-'0';ch=getchar();}
return c*f;
}
int main(){
int t;t=read();
while(t--){
n=read();k=read();
for(int i=0; i<n; ++i){
arr[i]=read();
}
LL ans=0;
while(!deq_max.empty()) deq_max.pop_back();
while(!deq_min.empty()) deq_min.pop_back();
for(i=0,j=0; i<n; ++i){
while(!deq_max.empty()&&deq_max.back()<arr[i]) deq_max.pop_back();deq_max.push_back(arr[i]);
while(!deq_min.empty()&&deq_min.back()>arr[i]) deq_min.pop_back();deq_min.push_back(arr[i]);
while(!deq_max.empty()&&!deq_min.empty()&&deq_max.front()-deq_min.front()>=k){
ans+=(i-j);
if(deq_max.front()==arr[j]) deq_max.pop_front();
if(deq_min.front()==arr[j]) deq_min.pop_front();
j++;
}
}
while(j<n){
ans+=(i-j);
j++;
}
printf("%I64d\n",ans);
} return 0;
}

最新文章

  1. Python文件读写
  2. HTML打折计算价格
  3. 无需FQ,自建本地CDN,秒上StackOverFlow!
  4. 转: KindEditor 图片空间文件增加删除文件、文件夹功能(ASP语言环境)
  5. SharePoint 2013 创建web应用程序报错&quot;This page can’t be displayed&quot;
  6. windows 环境下安装plpython语言环境到postgresql数据库
  7. 初学HTML5系列一:简单介绍
  8. Graphs and Minimum Cuts(Karger&#39;s Min-Cut Algorithm)
  9. 如何为ios酷我音乐盒下载导出的音乐文件(使用Java程序设计)
  10. 自动化辅助工具Firebug和Firepath的安装
  11. IDA分析脱壳后丢失导入表的PE
  12. pygame写贪吃蛇
  13. QT 设计师使用样式表添加背景
  14. 最小生成树之Kruskal(克鲁斯卡尔)算法
  15. python下的并发编程
  16. python多进程使用及线程池的使用方法
  17. 4.hadoop的安装与配置
  18. nomad 0.9 新特性
  19. Linux:从文件中搜索关键字并显示行数(cat,grep函数)
  20. HTTPS安全不?

热门文章

  1. 关于.Net的强名称(Strong Name)
  2. java虚拟机(九)--常用jvm参数
  3. vuex状态管理demo
  4. Luogu P3110 [USACO14DEC]驮运Piggy Back
  5. 为什么map对象不能使用stl中的sort函数
  6. 关于Linux字符集的查看及修改
  7. java 十六周总结
  8. Linux 复习二
  9. INT32 System_UserKeyFilter(NVTEVT evt, UINT32 paramNum, UINT32 *paramArray)
  10. 【15】AngularJS&#160;输入验证