随机数据下 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种获取随机数据的方法,并分析了各自的优缺点. ...
随机推荐
- blazor优雅的方式导入组件相关的js脚本
基本的组件导入方式为: 1 await JsRuntime.InvokeVoidAsync("import", $"XXXXX.js"); 优雅的组件导入方式: ...
- go-zero goctl命令图解
- 在IDEA中配置GitHub
目录 在IDEA中配置GitHub 安装GitHub插件 账号授权 方法一:点击Add Account... 添加账号 方法二:使用token登录 共享到GitHub 查看文件变更列表 add文件 移 ...
- 开发环境需要同时安装2个nodejs版本
由于同时有vue2和vue3的项目开发情况,vue2项目的nodejs版本是12,vue3项目在node12版本下运行不了,要求最低14版本,因此要用nvm同时安装和控制2个版本. 安装步骤: 1.卸 ...
- Chrome 插件 V3 版本 Manifest.json 中的内容脚本(Content Scripts)解析
内容脚本(Content Scripts) 指定在用户打开某些网页时要使用的 JavaScript 或 CSS 文件. 内容脚本是在网页环境中运行的文件.通过使用标准文档对象模型 (DOM),开发者能 ...
- [数字华容道] Html+css+js 实现小游戏
[数字华容道] Html+css+js 实现小游戏 效果图 代码预览 在线预览地址 代码示例 <!DOCTYPE html> <html> <head> <m ...
- uni-app写微信小程序,data字段循环引用
在写程序过程中,需要使用到 globalData里的内容,而这个全局变量,在uni-app上需要通过: var app=getApp(); app.globalData.xxx=xxx来使用. 我觉得 ...
- 【c++】const 限定符
#include <iostream> int main() { using namespace std; cout << "Hello World!\n" ...
- GeneralUpdate .Net5 WPF、Winfrom、控制台应用自动更新组件
https://www.bilibili.com/video/BV1aX4y137dd/?vd_source=43d3e66cc2ad46bac54dfb0c6a3a0a23 GeneralUpdat ...
- springboot~封装依赖引用包jar还是pom,哪种更规范
将多个第三方包封装成一个项目后,如果你的目的是让其他开发人员可以直接引用这些依赖,一般来说有两种常见的方式: 打成JAR包:将封装好的项目编译打包成JAR文件,其他开发人员可以将这个JAR文件添加到他 ...