题目

给定\(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. 矩池云教程|体验 OpenAI 最近推出的 3D 生成模型 Shap-E!

    Shap-E 是由 OpenAI 最近推出的3D生成模型,使用者可以通过简单的文字或图像进行三维模型的生成,OpenAI 认为相比于点云的生成模型Point-E,Shap-E的收敛更快.本文将展示如何 ...

  2. 第125篇: 期约Promise基本特性

    好家伙,本篇为<JS高级程序设计>第十章"期约与异步函数"学习笔记 1.非重入期约 1.1.可重入代码(百度百科) 先来了解一个概念 可重入代码(Reentry cod ...

  3. 【Azure 应用服务】Azure Function Timer触发函数加上Singleton后的问题

    问题描述 在Azure Function Timer Trigger的函数中,添加了Singleton属性,当Function的实例变为3个后,发现Timer函数并没有在三个实例上同时运行,每次触发时 ...

  4. C#多线程(5):资源池限制

    目录 Semaphore.SemaphoreSlim 类 Semaphore 类 示例 示例说明 信号量 SemaphoreSlim类 示例 区别 Semaphore.SemaphoreSlim 类 ...

  5. 15. JVM垃圾回收器详解

    1. 垃圾回收器的分类 和 GC性能指标 垃圾收集器没有在规范中进行过多的规定,可以由不同的厂商.不同版本的JVM来实现. 由于JDK的版本处于高速迭代过程中,因此Java发展至今已经衍生了众多的GC ...

  6. 当未指定且存在多个构造器,实例化对象时Spring如何选择?

    前言 在前面的讲解中,我们了解了如何获取构造器.当只有一个符合条件的构造器时,自然会选择它作为初始化的构造器.然而,在上一节中,我们遇到了一种特殊情况:当有多个符合条件的构造器时,返回的是一个数组.在 ...

  7. 如何避免MYSQL主从延迟带来的读写问题?

    在MYSQL 部署架构选型上,许多公司都会用到主从读写分离的架构,如下是一个一主一从的架构,主库master负责写入,从库slave进行读取. 但是既然是读写分离,必然会面临这样一个问题,当在主库上进 ...

  8. Ansible 基础入门

    2)Ansible 介绍 Ansible 基本概念 Ansible 是一种自动化运维工具,基于 Paramiko 开发的,并且基于模块化工作,Ansible 是一种集成 IT 系统的配置管理.应用部署 ...

  9. wav 格式音频文件生成例子

    wavfile is a simple sound library for use in CSE 20211. This library allows you to generate arbitrar ...

  10. 百度api经历

    底部参考文档,欢迎点击:https://www.runoob.com/http/http-content-type.html 这两天遇到了点糟心事,因为小伙伴走了.然后事情起因是这样的,来了个任务封装 ...