前言

这篇博客是根据我在打这道题的时候遇到的问题,来打的,有些细节可能考虑不到。

题目

在N(1<=N<=100000)个数A1…An组成的序列上进行M(1<=M<=100000)次操作,操作有两种:

(1)1 L R C:表示把A[L]到A[R]增加C(C的绝对值不超过10000);

(2)2 L R:询问A[L]到A[R]之间的最大值。

分析

由于本人刚刚学会splay,不够精通,splay的打法这里就先不说。

就讲讲求区间极值的方法吧。

对于每个位置开一个节点,记录这个节点的值、在以它为根的子树中的最大值。

当我们要对一个区间进行查询或修改时,

假设修改的区级为\([l,r]\)

我们将l-1转到根节点,r+1转到根节点的右儿子(为了保证有l-1和r+1节点,另外加入0和n+1节点)

那么根据二叉查找树的性质,

r+1的左子树就是要查询或修改的区间。



接着,还要处理lazy标记,

当我们再将x节点转到y节点的儿子时只需从y到x,将标记全部下传就可以了。

#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const int maxlongint=2147483647;
const int mo=1000000007;
const int N=100005;
using namespace std;
int l[N],r[N],lazy[N],mx[N],a[N],fa[N],root,q1,q2;
int tot,n,m,ans;
void clear()
{
fa[0]=lazy[0]=l[0]=r[0]=0;
mx[1]=mx[tot]=a[1]=a[tot]=mx[0]=a[0]=-maxlongint;
}
void doda(int x,int z)
{
mx[x]+=z;
a[x]+=z;
lazy[x]+=z;
}
void down(int x)
{
int z=lazy[x];
if(!z) return;
doda(l[x],z);
doda(r[x],z);
lazy[x]=0;
}
int getmax(int x)
{
mx[x]=max(mx[l[x]],mx[r[x]]);
mx[x]=max(mx[x],a[x]);
}
void zig(int x)
{
int y=fa[x];
fa[r[x]]=y;
if(l[fa[fa[x]]]==fa[x]) l[fa[y]]=x;
else r[fa[y]]=x;
l[y]=r[x];
r[x]=y;
fa[x]=fa[y];
fa[y]=x;
getmax(y);
getmax(x);
}
void zag(int x)
{
int y=fa[x];
fa[l[x]]=y;
if(l[fa[fa[x]]]==fa[x]) l[fa[y]]=x;
else r[fa[y]]=x;
r[y]=l[x];
l[x]=y;
fa[x]=fa[y];
fa[y]=x;
getmax(y);
getmax(x);
}
void splay(int p,int x)
{
if(x==p) return;
while(fa[x]!=p)
{
if(fa[fa[x]]==p)
{
if(l[fa[x]]==x) zig(x);
else zag(x);
break;
}
q1=(l[fa[fa[x]]]==fa[x]),q2=(l[fa[x]]==x);
if(q1)
{
if(q2) zig(fa[x]),zig(x);
else zag(x),zig(x);
}
else
{
if(q2) zig(x),zag(x);
else zag(fa[x]),zag(x);
}
}
}
void sola(int v,int x)
{
down(v);
if(x<v) sola(l[v],x);
if(v<x) sola(r[v],x);
getmax(v);
}
int get(int ll,int rr)
{
sola(root,ll-1);
clear();
splay(0,ll-1);
root=ll-1;
sola(root,rr+1);
clear();
splay(ll-1,rr+1);
return l[rr+1];
}
int main()
{
scanf("%d",&n);
r[1]=2;
fa[2]=1;
tot=2;
for(int i=1;i<=n;i++)
{
scanf("%d",&a[tot]);
mx[tot]=a[tot];
fa[tot+1]=tot;
r[tot]=++tot;
}
clear();
for(int i=tot;i>=1;i--) mx[i]=max(a[i],mx[r[i]]);
root=1;
scanf("%d",&m);
for(int i=1;i<=m;i++)
{
int x,ll,rr,c;
scanf("%d%d%d",&x,&ll,&rr);
ll++;
rr++;
clear();
if(x==1) scanf("%d",&c),doda(get(ll,rr),c);
else printf("%d\n",mx[get(ll,rr)]);
}
}

后记(2018.4.28)

明天就是GDOI2018了,回顾自己以前写的博客,看到我打splay居然zig、zag分开的之类的,回忆我多年来的OI历程,有些感慨。

总之,GDOI2018,加油。

伸展树splay之求区间极值的更多相关文章

  1. 伸展树(Splay tree)的基本操作与应用

    伸展树的基本操作与应用 [伸展树的基本操作] 伸展树是二叉查找树的一种改进,与二叉查找树一样,伸展树也具有有序性.即伸展树中的每一个节点 x 都满足:该节点左子树中的每一个元素都小于 x,而其右子树中 ...

  2. K:伸展树(splay tree)

      伸展树(Splay Tree),也叫分裂树,是一种二叉排序树,它能在O(lgN)内完成插入.查找和删除操作.在伸展树上的一般操作都基于伸展操作:假设想要对一个二叉查找树执行一系列的查找操作,为了使 ...

  3. 树-伸展树(Splay Tree)

    伸展树概念 伸展树(Splay Tree)是一种二叉排序树,它能在O(log n)内完成插入.查找和删除操作.它由Daniel Sleator和Robert Tarjan创造. (01) 伸展树属于二 ...

  4. 纸上谈兵: 伸展树 (splay tree)[转]

    作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢!  我们讨论过,树的搜索效率与树的深度有关.二叉搜索树的深度可能为n,这种情况下,每 ...

  5. 高级搜索树-伸展树(Splay Tree)

    目录 局部性 双层伸展 查找操作 插入操作 删除操作 性能分析 完整源码 与AVL树一样,伸展树(Splay Tree)也是平衡二叉搜索树的一致,伸展树无需时刻都严格保持整棵树的平衡,也不需要对基本的 ...

  6. 【BBST 之伸展树 (Splay Tree)】

    最近“hiho一下”出了平衡树专题,这周的Splay一直出现RE,应该删除操作指针没处理好,还没找出原因. 不过其他操作运行正常,尝试用它写了一道之前用set做的平衡树的题http://codefor ...

  7. [Splay伸展树]splay树入门级教程

    首先声明,本教程的对象是完全没有接触过splay的OIer,大牛请右上角.. 首先引入一下splay的概念,他的中文名是伸展树,意思差不多就是可以随意翻转的二叉树 PS:百度百科中伸展树读作:BoGa ...

  8. 伸展树Splay【非指针版】

    ·伸展树有以下基本操作(基于一道强大模板题:codevs维护队列): a[]读入的数组;id[]表示当前数组中的元素在树中节点的临时标号;fa[]当前节点的父节点的编号;c[][]类似于Trie,就是 ...

  9. POJ 3580 - SuperMemo - [伸展树splay]

    题目链接:http://poj.org/problem?id=3580 Your friend, Jackson is invited to a TV show called SuperMemo in ...

随机推荐

  1. ELK+Kafka

    kafka:接收java程序投递的消息的日志队列 logstash:日志解析,格式化数据为json并输出到es中 elasticsearch:实时搜索搜索引擎,存储数据 kibana:基于es的数据可 ...

  2. nginx源码包安装

    一.下载码源包 1. 获得源码包途径官方网站,可以获得最新的软件包 Nginx: www.nginx.org 2.具体实例展示(tengine) 下载源码包,准备软件包 准备编译环境如编译器gcc.m ...

  3. tensorflow 2.0 技巧 | 自定义tf.keras.Model的坑

    自定义tf.keras.Model需要注意的点 model.save() subclass Model 是不能直接save的,save成.h5,但是能够save_weights,或者save_form ...

  4. 结构体封装高精度 大整数BigInt

    曾经很讨厌高精度,因为它很长,不好记,而且在不是很单纯的题目里面感觉很烦(一个数就是一个数组).在一道题目中出现的时候总是用一些奇技淫巧混过去(比如把两个$long$ $long$拼在一起). 现在. ...

  5. c++ tcp 服务器和客户端例子

    目标:  完成一个精简TCP服务器,可接收来自多个用户的请求,并返回结果. 思路:  (1)服务器      C++ TCP服务器的实现主要由以下几个函数来完成:        a)socket    ...

  6. 思考-继续思考在数据库中两个表join的问题

    ##在资源有限的情况下,如何做两个大表的join? --- 假设系统资源:内存1G,大表10G,小表2G. --- 都拿到内存中进行笛卡尔集肯定不行,内存没有那么大. 最简单的办法是对两个表建索引,但 ...

  7. Java Android 开发数字不足位数前面补0

    import java.text.DecimalFormat; public void changeColor(View view) { DecimalFormat decimalFormat = n ...

  8. set的常见用法

    set的使用 set是什么 set是一个内部有序且不含重复元素的容器 用处 *使得元素自动有序 *去除重复元素 set的引入 # include <set> using namespace ...

  9. E - 卿学姐与城堡的墙(树状数组求逆序数)

    卿学姐与城堡的墙 Time Limit: 2000/1000MS (Java/Others)     Memory Limit: 65535/65535KB (Java/Others) Submit  ...

  10. Exchange 2010的部署

    实验拓扑: 实验准备条件:计算机基本配置的准备:DC (vbers):配置IP地址.子网掩码.网关EX (vbers2):配置IP地址.子网掩码.网关.指定DNS 域环境搭建的准备1.在计算机名为vb ...