P1198 [JSOI2008]最大数(区间最值 ST || 线段树 )

题目链接:

题目描述:
现在请求你维护一个数列,要求提供以下两种操作:
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;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值