题目描述:
现在请求你维护一个数列,要求提供以下两种操作:
1、 查询操作。
语法:Q L
功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值。
限制:LL不超过当前数列的长度。(L > 0)
2、 插入操作。
语法:A n
功能:将n加上t,其中tt是最近一次查询操作的答案(如果还未执行过查询操作,则t=0),并将所得结果对一个固定的常数D取模,将所得答案插入到数列的末尾。
限制:n是整数(可能为负数)并且在长整范围内。
注意:初始时数列是空的,没有一个数。
输入格式:
第一行两个整数,M和D,其中M表示操作的个数(M≤200,000),D如上文中所述,满足(0<D<2,000,000,000)
接下来的M行,每行一个字符串,描述一个具体的操作。语法如上文所述。
输出格式:
对于每一个查询操作,你应该按照顺序依次输出结果,每个结果占一行。
线段树解法报告:
区间最值问题,每当插入一个数就类似与单点修改,需要注意的是当q次全为修改操作,故区间大小应设为 m
线段树写法代码 :
#define first f
#define second s
#define ll long long
#define mp make_pair
#define pb push_back
#define pf push_front
#define lb lower_bound
#define ub upper_bound
#include <bits/stdc++.h>
#define pii pair<int,int>
#define mem(a,b) memset(a,b,sizeof(a))
using namespace std;
const int maxn=2e5+5;
const int MOD=1e9+7;
const int inf=1e9+7;
const double PI=acos(-1);
const double e=2.718281828459;
ll sum[maxn<<2];
void modify(int l,int r,int rt,int pos,ll val)
{
if(l==r){
sum[rt]=val;return ;
}
int mid=(l+r)>>1;
if(pos<=mid){
modify(l,mid,2*rt,pos,val);
}
else {
modify(mid+1,r,2*rt+1,pos,val);
}
sum[rt]=max(sum[2*rt],sum[2*rt+1]);
}
ll query(int L,int R,int l,int r,int rt)
{
if(L<=l&&R>=r){
return sum[rt];
}
int mid=(l+r)>>1;
ll ans=-1e18;
if(L<=mid){
ans=max(ans,query(L,R,l,mid,2*rt));
}
if(R>mid){
ans=max(ans,query(L,R,mid+1,r,2*rt+1));
}
return ans;
}
int main()
{
int n,cnt=0;
ll d,c,t=0;
char opt;
scanf("%d%lld",&n,&d);
for(int i=1;i<=n;i++){
scanf(" %c %lld",&opt,&c);
if(opt=='A'){
cnt++;
modify(1,n,1,cnt,(c+t)%d);
}
else{
t=query(cnt-c+1,cnt,1,n,1);
printf("%lld\n",t);
}
}
return 0;
}
ST表解题报告:
ST表也是用来解决区间RMQ问题的,p[i][j]表示区间[i,i+1,i+2…i+(1<<j)-1]的最值;当在尾部插入一个数,会发现它并不影响之前区间最值问题,而我们只需修改包含最后一个数的区间( O(log) )
ST表写法代码:
#define first f
#define second s
#define ll long long
#define mp make_pair
#define pb push_back
#define pf push_front
#define lb lower_bound
#define ub upper_bound
#include <bits/stdc++.h>
#define pii pair<int,int>
#define mem(a,b) memset(a,b,sizeof(a))
using namespace std;
const int maxn=2e5+5;
const int MOD=1e9+7;
const int inf=1e9+7;
const double PI=acos(-1);
const double e=2.718281828459;
int m[maxn];
int lg[maxn],p[maxn][20];
int main()
{
lg[1]=0;lg[2]=1;
for(int i=3;i<maxn;i++){
lg[i]=lg[i/2]+1;
}
int t=0,n,d,c,cnt=0;
char opt;
scanf("%d%d",&n,&d);
while(n--){
scanf(" %c %d",&opt,&c);
if(opt=='A'){
p[++cnt][0]=(c+t)%d;
for(int i=1;cnt-(1<<i)>=0;i++){
p[cnt-(1<<i)+1][i]=max(p[cnt-(1<<i)+1][i-1],p[cnt-(1<<(i-1))+1][i-1]);//修改以cnt为右端点,区间长度(1<<i)
}
}
else{
t=max(p[cnt-c+1][lg[c]],p[cnt-(1<<lg[c])+1][lg[c]]);
printf("%d\n",t);
}
}
return 0;
}