Treap学习基本入门

Treap标准学习模板

1.treap的基本了解

(1)splay与treap的区别

Splay的旋转操作是将普通节点转到根

而treap是将根转为普通节点

(2)treap的时空复杂度

treap的各项操作时间复杂度均摊为O(logn)。

由于treap的指针写法容易出错,所以通常用数组代替。

通常要有以下几种:

v[]——存放键值

rnd[]——存放随机出的优先级

l[],r[]——左右子树的下标

w[]——相同键值的个数

s[]——以该结点为祖先的节点个数(包括自己)

2.treap基本操作

左旋和右旋

inline void pushup(int k)
{s[k]=s[l[k]]+s[r[k]]+w[k];}
inline void rturn(int &k)
{int t=l[k];l[k]=r[t];r[t]=k;s[t]=s[k];pushup(k);k=t;}
inline void lturn(int &k)
{int t=r[k];r[k]=l[t];l[t]=k;s[t]=s[k];pushup(k);k=t;}

一定要明白平衡树通过维持树的平衡,来保证各操作的时间复杂度均摊较低。

插入

void ins(int &k,int num)
{
    if(!k)
    {k=++tot;v[k]=num;s[k]=w[k]=1;l[k]=r[k]=0;rnd[k]=rand();return;}
    s[k]++;
    if(v[k]==num)w[k]++;
    else if(num<v[k])
      {ins(l[k],num);if(rnd[l[k]]<rnd[k])rturn(k);}
    else
      {ins(r[k],num);if(rnd[r[k]]<rnd[k])lturn(k);}
}

插入操作其实是一个

*按照键值大小找寻插入位置

*插入后,随机优先级

*进行旋转,保证对于优先级来说,树满足堆的性质

的过程

删除

void del(int &k,int num)
{
    if(v[k]==num)
    {
        if(w[k]>1){w[k]--;s[k]--;}
        else if(l[k]*r[k]==0)k=l[k]+r[k];
        else if(rnd[l[k]]<rnd[r[k]])
         {rturn(k);del(k,num);}
        else {lturn(k);del(k,num);}
        return;
    }
    s[k]--;
    if(num<v[k])del(l[k],num);else del(r[k],num);
}

删除时:

*先找到待删结点

*只有一棵子树,直接用这棵子树代替这个待删结点

*有两棵子树,先将优先级高的那一棵旋转到根,然后递归在另一棵子树中删除结点

以上是最基本的操作,此外还可有寻找第k大,前驱,后继等功能。理解以上代码后,应该可以编写出。

3.treap入门例题

bzoj1503

http://www.lydsy.com/JudgeOnline/problem.php?id=1503

#include<iostream>
#include<cstdio>
#include<ctime>
#include<cstdlib>
using namespace std;
const int maxn=100020;
int s[maxn],l[maxn],r[maxn],v[maxn],tot=0,rnd[maxn],n,m,pre,root,ans;
inline void pushup(int k)
{s[k]=s[l[k]]+s[r[k]]+1;}
inline void rturn(int &k)
{int t=l[k];l[k]=r[t];r[t]=k;s[t]=s[k];pushup(k);k=t;}
inline void lturn(int &k)
{int t=r[k];r[k]=l[t];l[t]=k;s[t]=s[k];pushup(k);k=t;}
void ins(int &k,int num)
{
    if(!k)
    {k=++tot;v[k]=num;s[k]=1;l[k]=r[k]=0;rnd[k]=rand();return;}
  s[k]++;
  if(num<v[k])
  {ins(l[k],num);if(rnd[l[k]]<rnd[k])rturn(k);}
  else
    {ins(r[k],num);if(rnd[r[k]]<rnd[k])lturn(k);}
}
int del(int &k,int num){
    if(k==0)return 0;
    if(v[k]<num){
      int t=s[l[k]]+1;k=r[k];return t+del(k,num);
  }else{
    int t=del(l[k],num);s[k]-=t;return t;
    }
}
int find(int k,int x){
    if(s[l[k]]+1==x)return v[k]+pre;
    else if(s[l[k]]+1<x)return find(r[k],x-s[l[k]]-1);
    else return find(l[k],x);
}
int main(){
    srand(1);
    scanf("%d%d",&n,&m);
    char ch[1];int x;
    while(n--){
        scanf("%s%d",ch,&x);
        if(ch[0]==‘I‘)if(x>=m)ins(root,x-pre);
        if(ch[0]==‘A‘)pre+=x;
        if(ch[0]==‘S‘){
            pre-=x;ans+=del(root,m-pre);
      }
      if(ch[0]==‘F‘){
        if(x>s[root])printf("-1\n");
        else printf("%d\n",find(root,s[root]-x+1));
      }
  }
  printf("%d\n",ans);
  return 0;
}

poj2892

http://poj.org/problem?id=2892

#include<iostream>
#include<cstdio>
#include<ctime>
#include<cstdlib>
using namespace std;
const int maxn=80005;
int l[maxn],r[maxn],v[maxn],rnd[maxn],w[maxn],d[maxn];
int root,n,m,s,t,tot;
inline void rturn(int &k){int t=l[k];l[k]=r[t];r[t]=k;k=t;}
inline void lturn(int &k){int t=r[k];r[k]=l[t];l[t]=k;k=t;}
void ins(int &k,int num){
    if(k==0){
        k=++tot;w[k]=1;v[k]=num;l[k]=r[k]=0;rnd[k]=rand();return;
  }
  if(num==v[k])w[k]++;
  else if(num<v[k]){
      ins(l[k],num);if(rnd[l[k]]<rnd[k])rturn(k);
  }else{
      ins(r[k],num);if(rnd[r[k]]<rnd[k])lturn(k);
  }
}
void del(int &k,int num){
    if(v[k]==num){
        if(w[k]>1){w[k]--;return;}
        if(l[k]*r[k]==0)k=l[k]+r[k];
        else if(rnd[l[k]]<rnd[r[k]]){
            rturn(k);del(k,num);
      }else{
        lturn(k);del(k,num);
      }
      return;
  }
  if(num<v[k])del(l[k],num);else del(r[k],num);
}
void find(int &k,int num){
    if(k==0)return;
    if(v[k]>=num&&v[k]<t)t=v[k];
    if(v[k]<=num&&v[k]>s)s=v[k];
    if(v[k]<num)find(r[k],num);
    else find(l[k],num);
}
int main(){
    freopen("input.txt","r",stdin);
    srand(1);
    scanf("%d%d",&n,&m);
    char ch[1];int a;
    for(int i=1,j=0;i<=m;i++){
        scanf("%s",ch);
        if(ch[0]==‘D‘){
            scanf("%d",&a);
            ins(root,a);j++;d[j]=a;
      }
      if(ch[0]==‘R‘){del(root,d[j]);j--;}
      if(ch[0]==‘Q‘){
        scanf("%d",&a);
        s=0;t=n+1;
        find(root,a);
        if(s==a&&t==a)printf("0\n");
        else printf("%d\n",t-s-1);
      }
      for(int i=1;i<=tot;i++){}
    }
    return 0;
}

鸣谢提供模板的善良学长ZYF

鸣谢提供题目的善良学长Revain

时间: 2024-08-09 18:42:09

Treap学习基本入门的相关文章

Vue学习笔记入门篇——组件的使用

本文为转载,原文:Vue学习笔记入门篇--组件的使用 组件定义 组件 (Component) 是 Vue.js 最强大的功能之一.组件可以扩展 HTML 元素,封装可重用的代码.在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功能.在有些情况下,组件也可以是原生 HTML 元素的形式,以 is 特性扩展. 组件使用 注册 注册一个全局组件,你可以使用 Vue.component(tagName, options).组件在注册之后,便可以在父实例的模块中以自定义元素 的形式使用.

Vue学习笔记入门篇——组件的内容分发(slot)

本文为转载,原文:Vue学习笔记入门篇--组件的内容分发(slot) 介绍 为了让组件可以组合,我们需要一种方式来混合父组件的内容与子组件自己的模板.这个过程被称为 内容分发 (或 "transclusion" 如果你熟悉 Angular).Vue.js 实现了一个内容分发 API,使用特殊的 'slot' 元素作为原始内容的插槽. 编译作用域 在深入内容分发 API 之前,我们先明确内容在哪个作用域里编译.假定模板为: <child-component> {{ messa

Vue学习笔记入门篇——组件的通讯

本文为转载,原文:Vue学习笔记入门篇--组件的通讯 组件意味着协同工作,通常父子组件会是这样的关系:组件 A 在它的模版中使用了组件 B.它们之间必然需要相互通信:父组件要给子组件传递数据,子组件需要将它内部发生的事情告知给父组件.然而,在一个良好定义的接口中尽可能将父子组件解耦是很重要的.这保证了每个组件可以在相对隔离的环境中书写和理解,也大幅提高了组件的可维护性和可重用性.在 Vue 中,父子组件的关系可以总结为 props down, events up.父组件通过 props 向下传递

[IT学习]sql 入门及实例

sql 是一种数据库查询语言,可以让你很快的查询到数据.其实一般情况下,你也可以采用excel来查询数据库数据. 但是人们通常认为sql会更加灵活和方便一些. sql学习的入门网站: http://www.w3schools.com/SQl/sql_orderby.asp https://en.wikipedia.org/wiki/SQL sql 学习笔记: 1.如果你对结构化数据库有概念,那么还是很容易理解的.如果你对结构化数据库一点没有概念,请自行百度结构化数据库,对数据库结构.表.字段.查

storm学习之入门篇(一)

海量数据处理使用的大多是鼎鼎大名的hadoop或者hive,作为一个批处理系统,hadoop以其吞吐量大.自动容错等优点,在海量数据处理上得到了广泛的使用.但是,hadoop不擅长实时计算,因为它天然就是为批处理而生的,这也是业界一致的共识.否则最近这两年也不会有s4,storm,puma这些实时计算系统如雨后春笋般冒出来.先抛开s4,storm,puma这些系统不谈,我们首先来看一下,如果让我们自己设计一个实时计算系统,我们要解决哪些问题: 1.低延迟.都说了是实时计算系统了,延迟是一定要低的

PHP学习笔记——入门篇(1)——语法&变量

基础 PHP 语法 PHP 脚本可放置于文档中的任何位置. PHP 脚本以 <?php 开头,以 ?> 结尾: PHP 文件通常包含 HTML 标签以及一些 PHP 脚本代码. 注释:PHP 语句以分号结尾(;).PHP 代码块的关闭标签也会自动表明分号(因此在 PHP 代码块的最后一行不必使用分号). PHP 支持三种注释: //单行注释 #单行注释 /*多行注释*/ PHP 大小写敏感区分: 在 PHP 中,所有用户定义的函数.类和关键词(例如 if.else.echo 等等)都对大小写不

DOS命令学习(从入门到精通)

DOS命令学习 一.DOS使用常识 DOS(Disk Operating System)是一个使用得十分广泛的磁盘操作系统. 常见的DOS有两种:IBM公司的PC-DOS和微软公司的MS-DOS,它们的功能.命令用途格式都相同,我们常用的是MS-DOS. DOS的概况 DOS(Disk Operating System)是一个使用得十分广泛的磁盘操作系统,就连眼下流行的Windows9x/ME系统都是以它为基础. 常见的DOS有两种:IBM公司的PC-DOS和微软公司的MS-DOS,它们的功能.

MonoRail学习-入门实例篇

1.到官方网站下载安装文件,地址如下: http://www.castleproject.org/index.php/Castle:Download目前最新版本Beta5(您也可以不需要下载,直接使用实例代码中lib中的dll) 2.添加对Castle.MonoRail.Framework.dllCastle.MonoRail.Framework.Views.CompositeView.dllCastle.MonoRail.Framework.Views.NVelocity.dllNVeloci

[转载]MongoDB开发学习 经典入门

如果你从来没有接触MongoDB或对MongoDB有一点了解,如果你是C#开发人员,那么你不妨花几分钟看看本文.本文将一步一步带您轻松入门. 阅读目录 一:简介 二:特点 三:下载安装和开启服务器 四:使用mongo.exe 执行数据库增删改查操作 五:更多命令 六:MongoDB语法与现有关系型数据库SQL语法比较 七:可视化的客户端管理工具MongoVUE 八:在C#中使用官方驱动操作MongoDB 九,在C#中使用samus驱动操作MongoDB 十:写个批处理,方便开启Mongodb服务