题目

给定\(n\)个数,将这个数列复制\(k\)次得到数列\(a\),

对\(a\)满足区间赋值操作和区间最小值询问

\(n\leq 10^5,q\leq 10^5,k\leq 10^4即|a|\leq 10^9\)


分析

先考虑线段树的区间赋值和区间最小值询问,如果没有复制那就是基本操作,

考虑一个很大的变化就是不可能将整棵线段树完全建好,

考虑动态开点线段树,每当新开一个点时,就先赋值为最小值,

这样线段树的大小为\(O(q\log|a|)\),区间赋值和区间最小值照常完成,

所以问题就转换成快速求未改动时区间最小值,这个用RMQ做就可以了,

如果该区间长度不短于\(n\)就直接是原来\(n\)个数的最小值,

否则如果这一段跨越了复制点,就维护前缀最小值和后缀最小值拼凑,

如果区间在一段内直接用RMQ


代码

#include <cstdio>
#include <cctype>
#define rr register
using namespace std;
const int N=100011;
int f[N][17],two[17],lg[N],pre[N],suf[N],w[N<<6],lazy[N<<6],root,ls[N<<6],rs[N<<6],cnt,n,m;
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
inline void print(int ans){
if (ans>9) print(ans/10);
putchar(ans%10+48);
}
inline signed min(int a,int b){return a<b?a:b;}
inline signed RMQ(int l,int r){
rr int z=lg[r-l+1];
return min(f[l][z],f[r-two[z]+1][z]);
}
inline signed answ(int l,int r){
if (r-l+1>=n) return pre[n];
rr int posl=(l-1)/n+1,posr=(r-1)/n+1;
l-=(posl-1)*n,r-=(posr-1)*n;
return (posl==posr)?RMQ(l,r):min(suf[l],pre[r]);
}
inline void update(int &k,int l,int r,int x,int y,int z){
if (!k) w[k=++cnt]=answ(l,r);
if (l==x&&r==y) {w[k]=lazy[k]=z; return;}
rr int mid=(l+r)>>1;
if (lazy[k]){
if (!ls[k]) ls[k]=++cnt;
if (!rs[k]) rs[k]=++cnt;
w[ls[k]]=lazy[ls[k]]=lazy[k],
w[rs[k]]=lazy[rs[k]]=lazy[k],
lazy[k]=0;
}
if (y<=mid){
if (!rs[k]) w[rs[k]=++cnt]=answ(mid+1,r);
update(ls[k],l,mid,x,y,z);
}
else if (x>mid){
if (!ls[k]) w[ls[k]=++cnt]=answ(l,mid);
update(rs[k],mid+1,r,x,y,z);
}
else update(ls[k],l,mid,x,mid,z),update(rs[k],mid+1,r,mid+1,y,z);
w[k]=min(w[ls[k]],w[rs[k]]);
}
inline signed query(int &k,int l,int r,int x,int y){
if (!k) w[k=++cnt]=answ(l,r);
if (l==x&&r==y) return w[k];
rr int mid=(l+r)>>1;
if (lazy[k]){
if (!ls[k]) ls[k]=++cnt;
if (!rs[k]) rs[k]=++cnt;
w[ls[k]]=lazy[ls[k]]=lazy[k],
w[rs[k]]=lazy[rs[k]]=lazy[k],
lazy[k]=0;
}
if (y<=mid) return query(ls[k],l,mid,x,y);
else if (x>mid) return query(rs[k],mid+1,r,x,y);
else return min(query(ls[k],l,mid,x,mid),query(rs[k],mid+1,r,mid+1,y));
}
signed main(){
n=iut(),m=iut(),lg[0]=-1,two[0]=1,pre[0]=suf[n+1]=1e9+7;
for (rr int i=1;i<17;++i) two[i]=two[i-1]<<1;
for (rr int i=1;i<=n;++i) f[i][0]=iut(),lg[i]=lg[i>>1]+1;
for (rr int i=1;i<=n;++i) pre[i]=min(pre[i-1],f[i][0]);
for (rr int i=n;i>=1;--i) suf[i]=min(suf[i+1],f[i][0]);
for (rr int j=1;j<=lg[n];++j)
for (rr int i=1;i+two[j]-1<=n;++i)
f[i][j]=min(f[i][j-1],f[i+two[j-1]][j-1]);
for (rr int T=iut();T;--T){
rr int opt=iut(),l=iut(),r=iut();
if (opt==1) update(root,1,n*m,l,r,iut());
else print(query(root,1,n*m,l,r)),putchar(10);
}
return 0;
}

#RMQ,动态开点线段树#CF803G Periodic RMQ Problem的更多相关文章

  1. Codeforces 803G Periodic RMQ Problem ST表+动态开节点线段树

    思路: (我也不知道这是不是正解) ST表预处理出来原数列的两点之间的min 再搞一个动态开节点线段树 节点记录ans 和标记 lazy=-1 当前节点的ans可用  lazy=0 没被覆盖过 els ...

  2. [2016湖南长沙培训Day4][前鬼后鬼的守护 chen] (动态开点线段树+中位数 or 动规 or 贪心+堆优化)

    题目大意 给定一个长度为n的正整数序列,令修改一个数的代价为修改前后两个数的绝对值之差,求用最小代价将序列转换为不减序列. 其中,n满足小于500000,序列中的正整数小于10^9 题解(引自mzx神 ...

  3. [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...

  4. 【BZOJ-4636】蒟蒻的数列 动态开点线段树 ||(离散化) + 标记永久化

    4636: 蒟蒻的数列 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 247  Solved: 113[Submit][Status][Discuss ...

  5. codeforces 893F - Physical Education Lessons 动态开点线段树合并

    https://codeforces.com/contest/893/problem/F 题意: 给一个有根树, 多次查询,每次查询对于$x$i点的子树中,距离$x$小于等于$k$的所有点中权值最小的 ...

  6. codeforces 915E - Physical Education Lessons 动态开点线段树

    题意: 最大$10^9$的区间, $3*10^5$次区间修改,每次操作后求整个区间的和 题解: 裸的动态开点线段树,计算清楚数据范围是关键... 经过尝试 $2*10^7$会$MLE$ $10^7$会 ...

  7. CF915E Physical Education Lessons 动态开点线段树

    题目链接 CF915E Physical Education Lessons 题解 动态开点线段树 代码 /* 动态开点线段树 */ #include<cstdio> #include&l ...

  8. 洛谷P3313 [SDOI2014]旅行(树链剖分 动态开节点线段树)

    题意 题目链接 Sol 树链剖分板子 + 动态开节点线段树板子 #include<bits/stdc++.h> #define Pair pair<int, int> #def ...

  9. NOIP2017 列队——动态开点线段树

    Description: Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n×m名学生,方阵的行数为  ...

  10. 洛谷P3120 [USACO15FEB]牛跳房子(动态开节点线段树)

    题意 题目链接 Sol \(f[i][j]\)表示前\(i\)行\(j\)列的贡献,转移的时候枚举从哪里转移而来,复杂度\(O(n^4)\) 然后考虑每一行的贡献,动态开节点线段树维护一下每种颜色的答 ...

随机推荐

  1. 2021-10-11 vue的第三方组件二次封装

    原理 v-bind="$attrs"继承所有属性和props. v-on="$listeners"继承所有的方法. <template> <d ...

  2. django学习第六天---shell指令,单表基于双下划线的模糊查询,distinct注意点,字段的choices属性,url反向解析,orm多表操作创建表

    shell指令 命令 python manage.py shell 在Terminal,执行上面这个指令会进入到python解释器环境中,并且加载了我们当前django项目配置环境,所以可以在当前sh ...

  3. 我的第一个项目(十五) :完成数据保存功能(后端,改update)

    好家伙,   代码已开源(Gitee) PH-planewar: 个人开发的全栈小游戏 前端:vue2 + element-ui 后端: Springboot + mybatis-plus 数据库: ...

  4. 解密Spring中的Bean实例化:推断构造方法(上)

    在Spring中,一个bean需要通过实例化来获取一个对象,而实例化的过程涉及到构造方法的调用.本文将主要探讨简单的构造推断和实例化过程,让我们首先深入了解实例化的步骤. 实例化源码 protecte ...

  5. Jmeter 响应断言你知道多少?

    1 断言各组件介绍 Apply to:同上 测试字段: * 响应文本:响应体 * 响应代码:响应状态码 * 响应信息:状态码的消息 * 响应头:顾名思义就是响应头 * 请求头:顾名思义就是请求头 * ...

  6. Jmeter如何分布式执行脚本?

    Jmeter分布式执行原理: JMeter分布式执行时,选择其中一台作为调度机(master),其他机器作为执行机(slave): master会在本地编辑好jmx压测脚本,执行时,master将jm ...

  7. 十: SQL执行流程

    SQL执行流程 1. MySQL 中的 SQL执行流程 MySQL的查询流程: 1.1 查询缓存 Server 如果在查询缓存中发现了这条 SQL 语句,就会直接将结果返回给客户端:如果没 有,就进入 ...

  8. Android---Android 开发四大组件

    Android 应用程序组件 应用程序组件是一个Android应用程序的基本构建块.这些组件由应用清单文件松耦合的组织.AndroidManifest.xml描述了应用程序的每个组件,以及他们如何交互 ...

  9. ReentrantLock原理CAS+AQS队列

    ReentrantLock主要利用CAS+AQS队列来实现.它支持公平锁和非公平锁,两者的实现类似. CAS:Compare and Swap,比较并交换.CAS有3个操作数:内存值V.预期值A.要修 ...

  10. Python 潮流周刊第 41 期(摘要),赠书5本

    本周刊由 Python猫 出品,精心筛选国内外的 250+ 信息源,为你挑选最值得分享的文章.教程.开源项目.软件工具.播客和视频.热门话题等内容.愿景:帮助所有读者精进 Python 技术,并增长职 ...