首页 > 其他分享 >2024.4.5 RMQ补题

2024.4.5 RMQ补题

时间:2024-04-05 22:00:50浏览次数:30  
标签:ch 2024.4 int read RMQ maxn 补题 && getchar

P2048 [NOI2010] 超级钢琴

前缀和处理连续子段的和弦,然后 rmq 求最大值

运用堆存储最优答案

每次取出堆头统计一次后,除掉统计点再分成两段加入即可,共 k 次

#include<bits/stdc++.h>
#define maxn 500010
#define INF 0x3f3f3f3f
#define int long long

using namespace std;

int n,k,l,r,ans;
int f[maxn][30],sum[maxn];
struct node{
  int fr,to,l,r;
  bool operator < (const node &b) const{
    return sum[to]-sum[fr-1]<sum[b.to]-sum[b.fr-1];
  }
};

int read(){
  int s=0,w=1;char ch=getchar();
  while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}
  while(ch>='0'&&ch<='9')s=(s<<1)+(s<<3)+ch-'0',ch=getchar();
  return s*w;
}

int query(int l,int r){
  int k=log2(r-l+1);
  int x=f[l][k],y=f[r-(1<<k)+1][k];
  return sum[x]>sum[y]?x:y;
}

signed main(){
//  memset(f,128,sizeof f);
  n=read();k=read();l=read();r=read();
  for(int i=1;i<=n;i++) 
    sum[i]=sum[i-1]+read(),f[i][0]=i;
  int lenth=log2(n);
  for(int j=1;j<=lenth;j++)
    for(int i=1;i+(1<<j)-1<=n;i++){
      int x=f[i][j-1],y=f[i+(1<<(j-1))][j-1];
      f[i][j]=(sum[x]>sum[y])?x:y;
    }
  priority_queue<node> q;
  for(int i=1;i<=n;i++)
    if(i+l-1<=n) q.push((node){i,query(i+l-1,min(n,i+r-1)),i+l-1,min(n,i+r-1)});
  for(int i=1;i<=k;i++){
    node u=q.top();q.pop();
    ans+=sum[u.to]-sum[u.fr-1];
    if(u.l!=u.to) q.push((node){u.fr,query(u.l,u.to-1),u.l,u.to-1});
    if(u.r!=u.to) q.push((node){u.fr,query(u.to+1,u.r),u.to+1,u.r}); 
  }
  printf("%lld\n",ans);
  return 0;
}

UVA11235 Frequent values

非降序数列,因此可以直接统计数字的值和个数,即数字段的长度

再记录每一段的左右和每个点所在的段数

查 (l,r) 就可以查中间的若干段和两边的残段,其中中间用 st 表

#include <bits/stdc++.h>
#define maxn 100010
using namespace std;

int read(){
	int s=0,w=1;char ch=getchar();
	while(ch<'0'||ch>'9'){if(ch=='-') w=-1;ch=getchar();}
	while(ch>='0'&&ch<='9') s=(s<<1)+(s<<3)+ch-'0',ch=getchar();
	return s*w;
}

int n,q,le,ri,ans;
int a[maxn],num[maxn],l[maxn],r[maxn],d[maxn][30];
vector<int> cnt;

int query(int l,int r){
    int k=log2(r-l+1);
    return max(d[l][k],d[r-(1<<k)+1][k]);
}

int main(){
    while (scanf("%d",&n)){
    	if (!n) break;q=read();
        for (int i=0;i<n;++i) a[i]=read();
        a[n]=a[n-1]+1;int start=-1;
        for (int i=1;i<=n;++i){
            if (i==0||a[i]>a[i-1]){
                if (i>0) {
                    cnt.push_back(i-start);
                    for (int j=start; j<i; ++j) {
                        num[j]=cnt.size()-1;
                        l[j]=start;r[j]=i-1;
                    }
                }
                start=i;
            }
        }
        int _n=cnt.size();
        for (int i=0;i<_n;++i) d[i][0]=cnt[i];
        for (int j=1;(1<<j)<=_n;++j)
            for (int i=0;i+(1<<j)-1<_n;++i) 
                d[i][j]=max(d[i][j-1],d[i+(1<<(j-1))][j-1]);
        while (q--){
            le=read()-1;ri=read()-1;
            if (num[le]==num[ri]) ans=ri-le+1; 
			else {
                ans=max(ri-l[ri]+1,r[le]-le+1);
                if(num[le]+1<num[ri])
                    ans=max(ans, query(num[le]+1,num[ri]-1));
            }
            printf("%d\n", ans);
        }
	}
    return 0;
}

UVA11297Census

二维 rmq

再一个方向开个线段树,然后每个点为另一个方向开一个新的线段树

#include <bits/stdc++.h>  
using namespace std;  
  
int n,m,a[501][501],maxn,minn;  
struct node{ int maxv,minv;}p[2001][2001];  

int read(){
	int s=0,w=1;char ch=getchar();
	while(ch<'0'||ch>'9'){if(ch=='-') w=-1;ch=getchar();}
	while(ch>='0'&&ch<='9') s=(s<<1)+(s<<3)+ch-'0',ch=getchar();
	return s*w;
}
  
void build2(int mo,int ml,int mr,int o,int l,int r){  
    if(l!=r){  
        int m=(l+r)>>1;  
        build2(mo,ml,mr,o<<1,l,m);  
        build2(mo,ml,mr,(o<<1)+1,m+1,r);  
        p[mo][o].maxv=max(p[mo][o<<1].maxv,p[mo][(o<<1)+1].maxv);  
        p[mo][o].minv=min(p[mo][o<<1].minv,p[mo][(o<<1)+1].minv);  
    }  
    else{  
        if(ml==mr)p[mo][o].maxv=p[mo][o].minv=a[ml][l];  
        else {  
            p[mo][o].maxv=max(p[mo<<1][o].maxv,p[(mo<<1)+1][o].maxv);  
            p[mo][o].minv=min(p[mo<<1][o].minv,p[(mo<<1)+1][o].minv);  
        }  
    }  
}  
  
void build1(int o,int l,int r){  
    if(l!=r){  
        int m=(l+r)>>1;  
        build1(o<<1,l,m);  
        build1((o<<1)+1,m+1,r);  
    }  
    build2(o,l,r,1,1,n);  
}  
  
void query2(int o,int l,int r,int ql,int qr,int mo){  
    if(l>=ql && r<=qr){  
        maxn=max(maxn,p[mo][o].maxv);  
        minn=min(minn,p[mo][o].minv);  
    }  
    else{  
        int m=(l+r)>>1;  
        if(ql<=m)query2(o<<1,l,m,ql,qr,mo);  
        if(qr>m)query2((o<<1)+1,m+1,r,ql,qr,mo);  
    }  
}  
  
void query1(int o,int l,int r,int ql,int qr,int l2,int r2){  
    if(l>=ql && r<=qr){  
        query2(1,1,n,l2,r2,o);  
    }  
    else{  
        int m=(l+r)>>1;  
        if(ql<=m)query1(o<<1,l,m,ql,qr,l2,r2);  
        if(qr>m)query1((o<<1)+1,m+1,r,ql,qr,l2,r2);  
    }  
}  
  
void update2(int o,int l,int r,int q,int val,int mo){  
    if(l==r){  
        if(val==-1){  
            p[mo][o].maxv=max(p[mo<<1][o].maxv,p[(mo<<1)+1][o].maxv);  
            p[mo][o].minv=min(p[mo<<1][o].minv,p[(mo<<1)+1][o].minv);  
            return;  
        }  
        p[mo][o].maxv=p[mo][o].minv=val;  
        return;  
    }  
    int m=(l+r)>>1;  
    if(q<=m)update2(o<<1,l,m,q,val,mo);else update2((o<<1)+1,m+1,r,q,val,mo);  
    p[mo][o].maxv=max(p[mo][o<<1].maxv,p[mo][(o<<1)+1].maxv);  
    p[mo][o].minv=min(p[mo][o<<1].minv,p[mo][(o<<1)+1].minv);  
}  
  
void update1(int o,int l,int r,int qx,int qy,int val){  
    if(l==r) update2(1,1,n,qy,val,o);  
    else {  
        int m=(l+r)>>1;  
        if(qx<=m)update1(o<<1,l,m,qx,qy,val);
		else update1((o<<1)+1,m+1,r,qx,qy,val);  
        update2(1,1,n,qy,-1,o);  
    }  
}  
  
int main()  {  
    n=read(); 
    for(int i=1;i<=n;++i)
		for(int j=1;j<=n;++j)
			a[i][j]=read();  
    build1(1,1,n);m=read();char opt;  
    for(int i=1,x,y,c,xx,yy;i<=m;++i){  
        cin>>opt;
        if(opt=='q'){   
            x=read();y=read();xx=read();yy=read();  
            minn=0x3f3f3f3f;maxn=0;  
            query1(1,1,n,x,xx,y,yy);  
            printf("%d %d\n",maxn,minn);  
        }  
        else {   
            scanf("%d%d%d",&x,&y,&c);  
            update1(1,1,n,x,y,c);  
        }  
    }  
}  

标签:ch,2024.4,int,read,RMQ,maxn,补题,&&,getchar
From: https://www.cnblogs.com/KnightL/p/18116273

相关文章

  • [题解]ABC346 补题C~E
    想起上次的ABC346没打,刚才虚拟参赛打了A~D,E题思路有,但是实现方式没选好导致WA了,没能在赛时做出来。写下题解记录一下~C-Σ用求和公式先把\(1\simk\)的和求出来:\(\frac{k(k+1)}{2}\),然后对于\(A\)数组中的元素依次减去就行(注意相同元素不能减\(2\)次)点击查看代码#include<b......
  • cfGlobalRound24--BC补题
    B-Doremy'sPerfectMathClass思路:假设答案集合,在普遍的答案集合中找到特别之处.!!假设!!a1,a2,a3,a4就是某个集合S的答案。a2-a1=a1,即a2=2*a1.必然的..因为a2-a1<a2,结果又存在于S中,结果只能是a1.a3-a2=a1ora2?假如a3-a2=a2,则a3=2*a2=4*a1.那么a3-a1=3*a1..又3*a1>a2......
  • 1.5 警惕和揭秘伪创新(1)《软件方法》2024.4更新
    DDD领域驱动设计批评文集做强化自测题获得“软件方法建模师”称号《软件方法》各章合集1.5警惕和揭秘伪创新初中数学里要学习全等三角形、相似三角形、SSS、SAS……,到了高中以后学了正弦定理、余弦定理等解三角形的知识……就不会再回去用初中的方法解题了。但是,不是所......
  • 2024-4-4 分块补题
    P3203[HNOI2010]弹飞绵羊记录每个位置跳出当前块所需要的步数和跳出的位置。从后往前统计#include<bits/stdc++.h>#definemaxn200100usingnamespacestd;intn,m,len;intpos[maxn],k[maxn];intnxt[maxn],stp[maxn];structfk{intl,r;}a[maxn];intread(){......
  • 2024.4 做题记录
    299.CF1534ELostArray难崩。题意转化为每次翻转\(m\)个\(01\)序列的元素,要把全\(0\)翻成全\(1\)。不想分讨。考虑直接最短路求最小步数,转移就枚举选多少个原本已经有的数。交互就记录方案就行了。300.P9537[YsOI2023]CF1764B很棒的题。考察终态,可以发现最后输......
  • 2024/4/4 分块补题
    2024/4/4分块补题P3203[HNOI2010]弹飞绵羊分块跳跳跳,核心是每次跳出当前块,用\(to[i]\)表示跳到的位置。#include<bits/stdc++.h>usingnamespacestd;#defineldlongdoubletemplate<typenameT>inlineTread(){Tx=0;charch=getchar();boolfl=false;......
  • 2024.4 第一周做题记录
    \(2024.4.2\)CF1336DYuiandMahjongSet题意:初始有一个值域在\([1,n]\)的多重整数集\(S\),且每个元素重复次数最多为\(n\),定义\(\operatorname{triple}(S)\)表示\(S\)中相同无序三元组数量,\(\operatorname{straight}(S)\)表示\(S\)中连续整数的无序三元组数量,告诉......
  • 2024SMU蓝桥训练1补题
    B-航班时间思路:地理知识--时差计算-东加西减。此处去程和返程方向相反,时差相加必然抵消。那么就可以知道实际飞行时间ps:这题有点奇怪,本地跑不过样例,交上去是AC。本地跑过样例,交上去RE,WA。RE好像是因为输入的格式不够严格..D-飞机降落思路:全排列枚举ordfs--dfs类似之前电科......
  • #19 2024.4.3
    694.pjudge21633【PER#2】2048695.loj3483「USACO2021.2Platinum」CountingGraphs696.loj2468「2018集训队互测Day2」神秘货币史。697.cf1935fAndrey'sTree反思。考虑一个\(mx\rightarrowmx+1\)的构造。那么它挺赢的。考虑一些cornercase,即\(u=mx......
  • 随堂练习2024.4.3
    建立规则,仪式,流程,模式:  定义代码编写和审查的标准,确保开发质量。  实施敏捷开发的仪式,如日常站会,迭代评审和回顾会议,以提高团队协作和项目透明度。 建立清晰的开发流程和里程碑,确保项目按时推进。给好行为正面的反馈:  对于按时完成任务且代码质量高的开发人员......