随机数据下 Sqrt Tree 的平替实现
原理
在随机数据下,把原序列分成 \(\sqrt n\) 个块,维护每个块的前缀后缀最大值,那么,在随机询问下,对于在一个块中的询问,暴力查询。
复杂度
概率 $ n ^ {-\frac{1}{2}}$ ,复杂度 \(O(\sqrt n)\) ,均摊 \(O(1)\) 。
对于在不同块中的询问,对连续块询问以及散块前缀后缀最大值查询。
如何查询连续块
再用一遍上面的方法,对于块内最大值构成的序列二次分块,照相同办法处理,而这一次的连续块可以直接 \(O(\sqrt n )\) 暴力处理。
复杂度
随机数据下:
建表 : \(O(n)\)
查询 : \(O(1)\)
单点修改 :\(O(\sqrt n)\) (单点修改只需重建一个大小为 \(O(\sqrt n)\) 的表和一个块内前缀后缀最大值即可,而表可以暴力重建,块也只用扫一遍)
代码 (由乃救爷爷)
#include<bits/stdc++.h>
#define int unsigned long long
using namespace std;
const int maxn = 5e3 + 10;
const int warma = 5e3;
const int maxw = 70;
const int Warma = 62;
int lg[maxn];
int cntsum,n,q,cnt;
int pre[maxw][maxw],suf[maxw][maxw];
int bye[maxw][maxw];
int maxans[maxw][maxw];
int seed,sumcnt=0;
namespace GenHelper
{
unsigned z1,z2,z3,z4,b;
unsigned rand_()
{
b=((z1<<6)^z1)>>13;
z1=((z1&4294967294U)<<18)^b;
b=((z2<<2)^z2)>>27;
z2=((z2&4294967288U)<<2)^b;
b=((z3<<13)^z3)>>21;
z3=((z3&4294967280U)<<7)^b;
b=((z4<<3)^z4)>>12;
z4=((z4&4294967168U)<<13)^b;
return (z1^z2^z3^z4);
}
}
void srand(unsigned x)
{using namespace GenHelper;
z1=x; z2=(~x)^0x233333333U; z3=x^0x1234598766U; z4=(~x)+51;}
int read()
{
using namespace GenHelper;
int a=rand_()&32767;
int b=rand_()&32767;
return a*32768+b;
}
class chunking{
public:
int b[maxn];
int pre[maxn];
int suf[maxn];
inline void maintain();
inline int ask(int l,int r);
int length;
int anser;
}block[maxn];
inline void handle(){
int res1=1,res2=1;
for(int i=1;i<=cntsum;i++){
if(res1>Warma) res1=1,res2++;
bye[res1][res2]=block[i].anser;
res1++;
}
cnt=res2;
for(int i=1;i<=cnt;i++){
pre[1][i]=bye[1][i];
for(int j=2;j<=Warma;j++){
pre[j][i]=max(pre[j-1][i],bye[j][i]);
}
suf[Warma][i]=bye[Warma][i];
for(int j=Warma-1;j>=1;j--){
suf[j][i]=max(suf[j+1][i],bye[j][i]);
}
}
for(int i=1;i<=cnt;i++){
maxans[i][i]=0;
for(int j=1;j<=Warma;j++){
maxans[i][i]=max(maxans[i][i],bye[j][i]);
}
}
for(int i=1;i<=cnt;i++){
for(int j=i+1;j<=cnt;j++){
maxans[i][j]=max(maxans[i][j-1],maxans[j][j]);
}
}
}
inline int question(int l,int r){
int lc = l/Warma+1;
l%=Warma;
if(l==0) l=Warma,lc--;
int rc= r/Warma+1;
r%=Warma;
if(r==0) r=Warma,rc--;
if(lc==rc){
int res=0;
for(int i=l;i<=r;i++){
res=max(res,bye[i][lc]);
}
return res;
}
if(lc==rc+1){
return max(suf[l][lc],pre[r][rc]);
}
else
{
int res = maxans[lc+1][rc-1];
res=max(res,max(suf[l][lc],pre[r][rc]));
return res;
}
}
inline void chunking::maintain(){
anser=0;
pre[1]=b[1];
anser=max(anser,b[1]);
for(int i=2;i<=length;i++) pre[i]=max(pre[i-1],b[i]),anser=max(anser,b[i]);
suf[length]=b[length];
for(int i=length-1;i>=1;i--) suf[i]=max(suf[i+1],b[i]);
}
inline int chunking::ask(int l,int r){
int res=0;
for(int i=l;i<=r;i++) res=max(res,b[i]);
return res;
}
inline int query(int l,int r){
int lc=l/warma+1;
l%=warma;
if(l==0) lc--,l+=warma;
int rc=r/warma+1;
r%=warma;
if(r==0) rc--,r+=warma;
if(lc==rc){
return block[lc].ask(l,r);
}
if(lc+1==rc){
return max(block[lc].suf[l],block[rc].pre[r]);
}
else
{
int res=question(lc+1,rc-1);
res=max(res,max(block[lc].suf[l],block[rc].pre[r]));
return res;
}
}
inline void change(int x,int val){
int xc=x/warma+1;
x%=warma;
if(x==0) xc--,x+=warma;
block[xc].b[x]=val;
for(int i=xc;i<=xc;i++) block[i].maintain();
handle();
}
inline void inti(){
for(int i=1;i<=maxn-10;i++){
lg[i]=log2(i);
}
cin>>n>>q>>seed;
srand(seed);
//seed=read();
int res1=1,res2=1;
for(int i=1;i<=n;i++){
if(res1>warma) block[res2].length=res1-1,res1-=warma,res2++;
block[res2].b[res1]=read();
//cout<<chifan()<<' ';
res1++;
}
block[res2].length=res1-1;
cntsum=res2;
for(int i=1;i<=cntsum;i++) block[i].maintain();
handle();
}
int anser;
signed main(){
//freopen("5.in","r",stdin);
//freopen("5.out","w",stdout);
inti();
while(q--){
int op;
op = 1;
if(op==1){
int l,r;
l=read()%n+1;
r=read()%n+1;
if(l>r) swap(l,r);
//cout<<query(l,r)<<'\n';
anser+=query(l,r);
}
if(op==2){
int x,val;
x=read();
val=read();
change(x,val);
}
}
cout<<anser;
}
随机数据下 Sqrt Tree 的平替实现的更多相关文章
- [ML学习笔记] 决策树与随机森林(Decision Tree&Random Forest)
[ML学习笔记] 决策树与随机森林(Decision Tree&Random Forest) 决策树 决策树算法以树状结构表示数据分类的结果.每个决策点实现一个具有离散输出的测试函数,记为分支 ...
- 一个比较全面的java随机数据生成工具包
最近,由于一个项目的原因需要使用一些随机数据做测试,于是写了一个随机数据生成工具,ExtraRanom.可以看成是Java官方Random类的扩展,主要用于主要用于测试程序.生成密码.设计抽奖程序等情 ...
- 大数据下的数据分析平台架构zz
转自http://www.cnblogs.com/end/archive/2012/02/05/2339152.html 随着互联网.移动互联网和物联网的发展,谁也无法否认,我们已经切实地迎来了一个海 ...
- 随机数据生成与对拍【c++版,良心讲解】
10.7更新:见最下面 离NOIP2018没剩多长时间了,我突然发现我连对拍还不会,于是赶紧到网上找资料,找了半天发现了一个特别妙的程序,用c++写的! 不过先讲讲随机数据生成吧. 很简单,就是写一个 ...
- 不平衡数据下的机器学习方法简介 imbalanced time series classification
imbalanced time series classification http://www.vipzhuanli.com/pat/books/201510229367.5/2.html?page ...
- .NET使用Bogus生成大量随机数据
.NET如何生成大量随机数据 在演示Demo.数据库脱敏.性能测试中,有时需要生成大量随机数据.Bogus就是.NET中优秀的高性能.合理.支持多语言的随机数据生成库. Bogus的Github链接: ...
- python中faker模块:产生随机数据的模块
#pip install faker #产生各种随机数据的模块 想要运用更多的随机数据,可以百度查找下
- 使用vs2010生成SQL Server 随机数据
前几天做测试数据,偶然发现vs2010中有一个生成随机数据的功能,记录下来,方便以后使用,确实非常的好用灵活快捷. 为了简单扼要的说明,下面我用一个实例来说明如何快捷使用: 在VS2010创建数据库项 ...
- 【阿里云产品公测】大数据下精确快速搜索OpenSearch
[阿里云产品公测]大数据下精确快速搜索OpenSearch 作者:阿里云用户小柒2012 相信做过一两个项目的人都会遇到上级要求做一个类似百度或者谷歌的站内搜索功能.传统的sql查询只能使用like ...
- MySQL查询随机数据的4种方法和性能对比
从MySQL随机选取数据也是我们最常用的一种发发,其最简单的办法就是使用”ORDER BY RAND()”,本文介绍了包括ORDER BY RAND()的4种获取随机数据的方法,并分析了各自的优缺点. ...
随机推荐
- Typecho博客网站迁移:MySQL ➡️ MarialDB
目录 1. 引言 2. Typecho的自定义配置迁移 3. 数据库迁移:MySQL- > MarialDB 3.1 在原服务器中备份并导出数据库文件 3.2 将"backupdb.s ...
- RVM Ruby 版本管理器的删除 Gatling
参考: https://www.jianshu.com/p/aef65d0c03a4
- csapp-attacklab(完美解决版)
注意:必须阅读Writeup,否则根本看不懂这个lab要怎么做 实验前准备 1.在终端中输入./ctarget和./rtarget结果报错 百度后得知自学的同学需要在执行文件时加上-q参数,不发送结果 ...
- AIRIOT物联网低代码平台如何配置db-driver驱动?
设备接入能力包括接入驱动类型及接入数据量,性能方面需要考量数据采集的稳定性和驱动的丰富性等多个因素.用户在选择物联网平台时,往往存在一些误区,比如很关注平台支持的驱动数量,越多越好.市场上确有支持上千 ...
- c语言在Linux中的使用
gcc版本升级 如何验证gcc正常使用,编译c以及运行 过程 要验证GCC(GNU Compiler Collection)是否正常使用,您可以按照以下步骤进行操作: 检查GCC是否安装:打开终端或命 ...
- jenkens
[root@mcw01 ~]$ ls .jenkins/ config.xml jenkins.install.UpgradeWizard.state nodeMonitors.xml secret. ...
- 上位机开发福利!快速掌握.NET中的Modbus通信
安装nuget包 Wesky.Net.OpenTools 1.0.8或以上版本.支持.net framework 4.6以上版本,以及所有.net core以及以上版本引用. 开发一个简单的Winf ...
- wblockCloneObjects 写块克隆的使用
写块克隆可以把当前数据库的实体写入到另一个dwg文件中去.用法根deepclone类似,不过deepclone只能复制到同一数据库中,而写块克隆是在不同数据库中进行复制的.写块克隆也算是深度克隆,能把 ...
- Linux之kill命令
1.kill命令的使用格式 kill [参数] [进程号] 2.kill命令的功能 发送指定的信号到相应进程.不指定型号将发送SIGTERM(15)终止指定进程.如果任然无法终止该程序可用" ...
- c#动态执行脚本的3种方式详解
1.使用Roslyn编译器 2.使用IronPython或IronRuby 3.使用JavaScript引擎 在C#中,可以使用一些第三方库或内置类库实现动态执行脚本的功能.以下是几个常用的方案: 1 ...