非常裸的一棵Splay树,需要询问的是区间gcd,但是区间上每个数分成了两种状态,做的时候分别存在val[2]的数组里就好。区间gcd的时候基本上不支持区间的操作了吧。。不然你一个区间里加一个数gcd都不知道怎么维护了,所以维护点的gcd是比较简单的,题目存在删除和增加,所以Splay树无误了。删除一个结点或区间的方法是先把它get出来,然后令root->ch[1]->ch[0]=null,记得每次操作完要提根。

#pragma warning(disable:4996)
#include<cstring>
#include<string>
#include<cstdio>
#include<iostream>
#define ll long long
#define maxn 401000
using namespace std; int n, q; int gcd(int a, int b)
{
while (a&&b) { a %= b; a ^= b; b ^= a; a ^= b; }
return a + b;
} struct Node
{
Node *ch[2], *p;
int size, val[2], gval[2];
int status;
Node(){
size = 0; val[0] = val[1] = 0; status = 0; gval[0] = gval[1] = 0;
}
bool d() {
return this == p->ch[1];
}
void setc(Node *c, int d){
ch[d] = c; c->p = this;
}
void cvalIt(int cval){
val[status] = cval;
gval[status] = gcd(val[status],gcd(ch[0]->gval[status], ch[1]->gval[status]));
}
void cstaIt(){
swap(val[status ^ 1], val[status]);
gval[status] = gcd(ch[0]->gval[status], ch[1]->gval[status]);
gval[status ^ 1] = gcd(gval[status ^ 1], gcd(ch[0]->gval[status ^ 1], ch[1]->gval[status ^ 1]));
status ^= 1;
}
void upd(){
size = ch[0]->size + ch[1]->size + 1;
for (int i = 0; i < 2; i++){
gval[i] = gcd(gcd(ch[0]->gval[i], ch[1]->gval[i]), val[i]);
}
}
}Tnull,*null=&Tnull; Node mem[maxn], *C = mem; Node*make(int v, int sta){
C->ch[0] = C->ch[1] = null;
C->size = 1;
C->val[sta] = v; C->val[sta ^ 1] = 0;
C->gval[sta] = v; C->gval[sta ^ 1] = 0;
C->status = sta;
return C++;
} int ax[maxn], bx[maxn]; Node*build(int l, int r)
{
if (l >= r) return null;
int m = (l + r) >> 1;
Node *t = make(ax[m], bx[m]);
t->setc(build(l, m), 0);
t->setc(build(m + 1, r), 1);
t->upd();
return t;
} Node *root;
void rot(Node*t) {
Node*p = t->p;
//p->relax();
//t->relax();
int d = t->d();
p->p->setc(t, p->d());
p->setc(t->ch[!d], d);
t->setc(p, !d);
p->upd();
if (p == root)
root = t;
} void splay(Node*t, Node*f = null) {
while (t->p != f) {
if (t->p->p == f)
rot(t);
else
t->d() == t->p->d() ? (rot(t->p), rot(t)) : (rot(t), rot(t));
}
t->upd();
} Node* select(int k) {
for (Node*t = root;;) {
//t->relax();
int c = t->ch[0]->size;
if (k == c)
return t;
if (k > c)
k -= c + 1, t = t->ch[1];
else
t = t->ch[0];
}
} Node*&get(int l, int r) { //[l,r)
Node*L = select(l - 1);
Node*R = select(r);
splay(L);
splay(R, L);
return R->ch[0];
} void print(Node* x)
{
if (x == null) return;
print(x->ch[0]);
printf(" %d", x->val[0]);
print(x->ch[1]);
} int main()
{
while (~scanf("%d%d", &n, &q))
{
for (int i = 1; i <= n; i++){
scanf("%d%d", &ax[i], &bx[i]);
}
C = mem;
ax[0] = ax[n + 1] = 0; bx[0] = bx[n + 1] = 0;
root = build(0, n + 2); root->p = null;
char type[3];
int li, ri, si,vi;
for (int i = 0; i < q; i++){
scanf("%s", type);
if (type[0] == 'Q'){
scanf("%d%d%d", &li, &ri, &si);
Node* &t=get(li, ri+1);
if (t->gval[si]) {
printf("%d\n", t->gval[si]);
}
else puts("-1");
}
else if (type[0] == 'I'){
scanf("%d%d%d", &li, &vi, &si);
Node *&t = get(li, li + 1);
t->ch[1] = make(vi, si);
splay(t);
}
else if (type[0] == 'D' ){
scanf("%d", &li);
Node *&t = get(li, li+1);
root->ch[1]->ch[0] = null;
splay(root->ch[1]);
}
else if (type[0] == 'R'){
scanf("%d", &li);
Node *&t = get(li, li + 1);
t->cstaIt();
splay(t);
}
else if (type[0] == 'M'){
scanf("%d%d", &li, &vi);
Node *&t = get(li, li + 1);
t->cvalIt(vi);
splay(t);
}
}
}
return 0;
}

ZOJ3765 Lights Splay树的更多相关文章

  1. Splay树-Codevs 1296 营业额统计

    Codevs 1296 营业额统计 题目描述 Description Tiger最近被公司升任为营业部经理,他上任后接受公司交给的第一项任务便是统计并分析公司成立以来的营业情况. Tiger拿出了公司 ...

  2. Splay树再学习

    队友最近可能在学Splay,然后让我敲下HDU1754的题,其实是很裸的一个线段树,不过用下Splay也无妨,他说他双旋超时,单旋过了,所以我就敲来看下.但是之前写的那个Splay越发的觉得不能看,所 ...

  3. 暑假学习日记:Splay树

    从昨天开始我就想学这个伸展树了,今天花了一个上午2个多小时加下午2个多小时,学习了一下伸展树(Splay树),学习的时候主要是看别人博客啦~发现下面这个博客挺不错的http://zakir.is-pr ...

  4. 1439. Battle with You-Know-Who(splay树)

    1439 路漫漫其修远兮~ 手抄一枚splay树 长长的模版.. 关于spaly树的讲解   网上很多随手贴一篇 貌似这题可以用什么bst啦 堆啦 平衡树啦 等等 这些本质都是有共同点的 查找.删除特 ...

  5. 伸展树(Splay树)的简要操作

    伸展树(splay树),是二叉排序树的一种.[两个月之前写过,今天突然想写个博客...] 伸展树和一般的二叉排序树不同的是,在每次执行完插入.查询.删除等操作后,都会自动平衡这棵树.(说是自动,也就是 ...

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

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

  7. hdu 3436 splay树+离散化*

    Queue-jumpers Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) To ...

  8. hdu 1890 splay树

    Robotic Sort Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tot ...

  9. hdu3487 splay树

    Play with Chain Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

随机推荐

  1. c/c++面试总结(2)

    4.深拷贝和浅拷贝 (1)什么时候会用到拷贝函数 一个对象以值传递的方式传入函数(就是作为入参) 一个对象以值传递的方式从函数返回(就是作为返回值) 一个对象需要通过另外一个对象进行初始化 (2)是否 ...

  2. virtualbox安装增强包及配置共享文件夹

       因为需要在host及虚拟机间传输数据,想使用共享文件夹.但是单独设置了共享文件夹后在centos里找不到共享文件夹,看了下要安装增强包.好吧,顺 便也解决下鼠标切换的问题,省的老是按右CTL切换 ...

  3. Linux下解决apache 报 403 forbidden 错

    三步搞定: 1. 打开终端 2. 输入 chcon -R -t httpd_sys_content_t /var/www/html # 后面的/var/www/html是网站的默认目录,可以根据自己的 ...

  4. sql存在一个表而不在另一个表中的数据

    (转)A.B两表,找出ID字段中,存在A表,但是不存在B表的数据.A表总共13w数据,去重后大约3W条数据,B表有2W条数据,且B表的ID字段有索引. 方法一 使用 not in ,容易理解,效率低  ...

  5. Ubuntu搜狗输入法的使用

    下载搜狗拼音输入法安装包 删除ibus,确认没有以ibus开头的包存在 sudo dpkg -P ibus ibus-gtk ibus-gtk3 ibus-pinyin ibus-pinyin-db- ...

  6. win8.1环境下硬盘安装centos6.5双系统

    作为一名软件工程师或是相关行业的从业人员,经常会用到linux系统,这里为大家介绍的安装linux的方案是在原有win环境下安装linux双系统,两个系统选择分别为win8.1 和centos6.5. ...

  7. [大牛翻译系列]Hadoop(19)MapReduce 文件处理:基于压缩的高效存储(二)

    5.2 基于压缩的高效存储(续) (仅包括技术27) 技术27 在MapReduce,Hive和Pig中使用可分块的LZOP 如果一个文本文件即使经过压缩后仍然比HDFS的块的大小要大,就需要考虑选择 ...

  8. SpringUtil

    /** SpringUtil.java {{IS_NOTE Purpose: Description: History: Thu Jun 1 13:53:53 2006, Created by hen ...

  9. eclipse 安装插件不生效

    操作: 下载了一个插件,插件中只有一个jar包,解压放到plugins目录,插件没生效. 解决方法: 把jar包直接扔到dropins目录下,并且删除eclipse\configuration\org ...

  10. Oracle 中的replace函数的应用

    replace 函数用法如下: replace('将要更改的字符串','被替换掉的字符串','替换字符串') oracle 中chr()函数 CHR() --将ASCII码转换为字符 语法CHR(nu ...