#RMQ,动态开点线段树#CF803G Periodic RMQ Problem
题目
给定\(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的更多相关文章
- Codeforces 803G Periodic RMQ Problem ST表+动态开节点线段树
思路: (我也不知道这是不是正解) ST表预处理出来原数列的两点之间的min 再搞一个动态开节点线段树 节点记录ans 和标记 lazy=-1 当前节点的ans可用 lazy=0 没被覆盖过 els ...
- [2016湖南长沙培训Day4][前鬼后鬼的守护 chen] (动态开点线段树+中位数 or 动规 or 贪心+堆优化)
题目大意 给定一个长度为n的正整数序列,令修改一个数的代价为修改前后两个数的绝对值之差,求用最小代价将序列转换为不减序列. 其中,n满足小于500000,序列中的正整数小于10^9 题解(引自mzx神 ...
- [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...
- 【BZOJ-4636】蒟蒻的数列 动态开点线段树 ||(离散化) + 标记永久化
4636: 蒟蒻的数列 Time Limit: 30 Sec Memory Limit: 256 MBSubmit: 247 Solved: 113[Submit][Status][Discuss ...
- codeforces 893F - Physical Education Lessons 动态开点线段树合并
https://codeforces.com/contest/893/problem/F 题意: 给一个有根树, 多次查询,每次查询对于$x$i点的子树中,距离$x$小于等于$k$的所有点中权值最小的 ...
- codeforces 915E - Physical Education Lessons 动态开点线段树
题意: 最大$10^9$的区间, $3*10^5$次区间修改,每次操作后求整个区间的和 题解: 裸的动态开点线段树,计算清楚数据范围是关键... 经过尝试 $2*10^7$会$MLE$ $10^7$会 ...
- CF915E Physical Education Lessons 动态开点线段树
题目链接 CF915E Physical Education Lessons 题解 动态开点线段树 代码 /* 动态开点线段树 */ #include<cstdio> #include&l ...
- 洛谷P3313 [SDOI2014]旅行(树链剖分 动态开节点线段树)
题意 题目链接 Sol 树链剖分板子 + 动态开节点线段树板子 #include<bits/stdc++.h> #define Pair pair<int, int> #def ...
- NOIP2017 列队——动态开点线段树
Description: Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n×m名学生,方阵的行数为 ...
- 洛谷P3120 [USACO15FEB]牛跳房子(动态开节点线段树)
题意 题目链接 Sol \(f[i][j]\)表示前\(i\)行\(j\)列的贡献,转移的时候枚举从哪里转移而来,复杂度\(O(n^4)\) 然后考虑每一行的贡献,动态开节点线段树维护一下每种颜色的答 ...
随机推荐
- 海康摄像SDK开发笔记(一):海康威视网络摄像头SDK介绍与模块功能
前言 视频监控.人脸识别等应用中经常使用到摄像头,当前占据主流视频监控摄像头就是海康和大华两家,都可通过自家的sdk或者是onvif方式使用和控制摄像头. 本文章讲解海康的sdk方式. 海康 ...
- django中使用celery异步发送邮件
申请163网易发送邮件权限 在django中settings配置文件 #配置邮件服务器 EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBac ...
- 【LeetCode数组#1二分法】二分查找、搜索插入、在排序数组中查找元素的第一个和最后一个位置
二分查找 题目 力扣704题目链接 给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 ...
- 【Azure Redis 缓存】使用StackExchange.Redis,偶发ERROR - Timeout performing HSET (15000ms)
问题描述 使用StackExchange.Redis 作为Redis客户端SDK,连接Azure Redis服务,长期运行后发现,每天都偶发 Timeout Error. 错误消息如下: StackE ...
- spring源码手写aop
AOP: aop切面编程,其实就是spring增强器的一个扩展,就是通过beanPostProcessor的after后置方式实现的,其中在after中把需要的bean通过放射+动态代理完 ...
- 解决 Genymotion 显示‘unable to start the virtual device’的问题
·解决方案 以管理员身份运行以下命令: bcdedit /set hypervisorlaunchtype off ,然后重启电脑,打开模拟器即可. 注意,一定是以[管理员]的身份运行的[命令提示符] ...
- 基于STM32F407MAC与DP83848实现以太网通讯一(STM32以太网(ETH)外设)
STM32F4xx 可以通过以太网按照 IEEE 802.3-2002 标准发送和接收数据.支持与外部物理层 (PHY) 相连的两个工业标准接口:默认情况下使用的介质独立接口 (MII)(在 IEEE ...
- coast 海岸 单词记忆方法
coast 海岸 单词记忆方法 coa 扣 想象一个碗扣下去 st站 碗的边和地面的接触面 就是海岸的边 coast 逼近的地方-海岸 coast (n.) - "margin of the ...
- crypto-js DES加密 base64 post传输
演示地址: http://pengchenggang.gitee.io/crypto-js-des/ 开源地址: https://gitee.com/pengchenggang/crypto-js-d ...
- .NET数据库ORM框架SqlSugar使用笔记
官方介绍 以下介绍来自SqlSugar官网: 高性能 ,不夸张的说,去掉Sql在数据库执行的时间,SqlSugar是EF数倍性能,另外在批量操作和一对多查询上也有不错的SQL优化 高扩展性 ,支持自定 ...