【题目链接】:http://hihocoder.com/problemset/problem/1333

【题意】

【题解】



伸展树;

要求提供操作:

1.插入一个元素,两个权值,id作为查找的比较权值,val作为储存信息;

2.将id在[a..b]范围内的点的val值改变d;d能为负值

3.将id在[a..b]范围内的点全部删掉;

4.查询id在[a..b]范围内的点的val值的和;

区间的更改要用到懒惰标记;

权值和,则需要在node域里面加一个变量,存当前子树的val和;

在旋转,插入,删除的时候,要更新权值和,同时下传懒惰标记.



【Number Of WA】



14(懒惰标记往下传的时候,乘错了,应该是乘子树的大小的)



【完整代码】

#include <bits/stdc++.h>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define LL long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define mp make_pair
#define pb push_back
#define fi first
#define se second
#define ms(x,y) memset(x,y,sizeof x) typedef pair<int,int> pii;
typedef pair<LL,LL> pll; const int dx[9] = {0,1,-1,0,0,-1,-1,1,1};
const int dy[9] = {0,0,0,-1,1,-1,1,-1,1};
const double pi = acos(-1.0);
const int N = 110; struct node
{
int id;
LL val,sumval,lazy_tag,siz;
node *par,*child[2];
node(int id,LL val): id(id),val(val),sumval(val),lazy_tag(0),siz(1),par(NULL){}
}; int n;
char s[3];
node *root; void up_data(node *v)//更新这个节点,更新它的大小和权值和
{
v->sumval = v->val,v->siz = 1;
rep1(i,0,1)
if (v->child[i]!=NULL)
{
node *y = v->child[i];
v->sumval+=y->sumval;
v->siz+=y->siz;
}
} void add_tag(node *x)//懒惰标记下传
{
if (x->lazy_tag==0) return;
rep1(i,0,1)
if (x->child[i]!=NULL)
{
node *v = x->child[i];
v->val+=x->lazy_tag;
v->sumval+=1LL*x->lazy_tag*v->siz;//注意乘的是儿子节点的子树大小
v->lazy_tag+=x->lazy_tag;
}
x->lazy_tag = 0;
} void rotate(node* const x, int c) {//注意懒惰标记的下传位置
node* const y = x->par;
add_tag(y),add_tag(x);
y->child[c ^ 1] = x->child[c];
if (x->child[c] != NULL) x->child[c]->par = y;
x->par = y->par;
if (y->par != NULL && y->par->child[0] == y) y->par->child[0] = x;
else if (y->par != NULL && y->par->child[1] == y) y->par->child[1] = x;
y->par = x;
x->child[c] = y;
up_data(y),up_data(x);
} inline bool _splay_parent(node *x, node* (&y), node* stop) {
return (y = x->par) != stop && (x == y->child[0] || x == y->child[1]);
} void splay(node* const x, node* const stop) {
for (node *y, *z; _splay_parent(x, y, stop); ) {
if (_splay_parent(y, z, stop)) {
const int c = y == z->child[0];
if (x == y->child[c]) rotate(x, c ^ 1), rotate(x, c);
else rotate(y, c), rotate(x, c);
} else {
rotate(x, x == y->child[0]);
break;
}
}
if (stop == NULL) root = x;
} node *cr(node *u,int id,int val)//返回插入的节点所在的位置,如果已经存在则返回那个节点
{
add_tag(u);
if (u->id==id) return u;
if (u->child[id>u->id]==NULL)
{
node *v = new node(id,val);
v->child[0] = v->child[1] = NULL;
v->par = u;
u->child[id>u->id] = v;
return v;
}
node *temp = cr(u->child[id>u->id],id,val);
up_data(u);
return temp;
} void cr(int id,int val)//插入某个节点
{
node *u = cr(root,id,val);
splay(u,NULL);
} node * cz(node *u,int id)//查找某个权值,不存在就返回空
{
if (u->id == id) return u;
if (u->child[id>u->id]==NULL) return NULL;
return cz(u->child[id>u->id],id);
} node *get_max(node * v)//获取子树最大的权值所在节点
{
if (v->child[1]==NULL)
return v;
return get_max(v->child[1]);
} node *get_min(node *v)//获取子树最小的权值所在节点
{
if (v->child[0]==NULL)
return v;
return get_min(v->child[0]);
} void sc(int l,int r)//删除[l..r]这个区间
{
cr(l,0),cr(r,0);//删除操作获取l..r的时候,是先找到l的前驱和r的后继,然后一起删掉 node * u = cz(root,l);
splay(u,NULL);
node *lu = get_max(u->child[0]); node * v = cz(root,r);
splay(v,NULL);
node *rv = get_min(v->child[1]); splay(lu,NULL);
splay(rv,lu);
rv->child[0] = NULL;
up_data(rv),up_data(lu);
} void gb(int a,int b,int c)//把区间[l..r]的值都改变c
{
node *u = cz(root,a-1),*v = cz(root,b+1);//同理插入一个a-1和一个b+1
bool ju1 = (u==NULL),ju2 = (v==NULL);
if (ju1) u = cr(root,a-1,0);
if (ju2) v = cr(root,b+1,0);
splay(u,NULL);
splay(v,u);
if (v->child[0]!=NULL)
{
node *y = v->child[0];
y->val+=c;
y->sumval+=1LL*y->siz*c;
y->lazy_tag+=c;
up_data(v),up_data(u);
}
if (ju1) sc(a-1,a-1);
if (ju2) sc(b+1,b+1);
} LL query(int a,int b)//询问区间[a..b]的权值和
{
node *u = cz(root,a-1),*v = cz(root,b+1);
bool ju1 = (u==NULL),ju2 = (v==NULL);//看看有没有a-1和b+1,如果有的话,就能方便获取[a..b]了
if (ju1) u = cr(root,a-1,0);//没有的话就自己插入一个
if (ju2) v = cr(root,b+1,0);
splay(u,NULL);
splay(v,u);
LL temp = v->child[0]->sumval;
if (ju1) sc(a-1,a-1);
if (ju2) sc(b+1,b+1);
return temp;
} int main()
{
ios::sync_with_stdio(false),cin.tie(0);//scanf,puts,printf not use
root = new node(-1,0);
root->child[0] = root->child[1] = NULL;
cr(21e8,0);
cin >> n;
rep1(i,1,n)
{
cin >> s;
if (s[0]=='I')
{
int id,val;
cin >> id >> val;
cr(id,val);
}
if (s[0]=='D')
{
int a,b;
cin >> a >> b;
sc(a,b);
}
if (s[0]=='M')
{
int a,b,c;
cin >> a >> b >> c;
gb(a,b,c);
}
if (s[0]=='Q')
{
int a,b;
cin >> a >> b;
cout << query(a,b) << endl;
} }
return 0;
}

【hihocoder 1333】平衡树·Splay2的更多相关文章

  1. hihocoder#1333 : 平衡树·Splay2 (区间操作)

    题面: #1333 : 平衡树·Splay2 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:好麻烦啊~~~~~ 小Hi:小Ho你在干嘛呢? 小Ho:我在干活啊! ...

  2. Hihocoder #1333 : 平衡树·Splay2

    1333 : 平衡树·Splay2 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:好麻烦啊~ 小Hi:小Ho你在干嘛呢? 小Ho:我在干活啊!前几天老师让我帮忙 ...

  3. hihocoder #1333 : 平衡树·Splay2

    描述 小Ho:好麻烦啊~~~~~ 小Hi:小Ho你在干嘛呢? 小Ho:我在干活啊!前几天老师让我帮忙管理一下团队的人员,但是感觉好难啊. 小Hi:说来听听? 小Ho:事情是这样的.我们有一个运动同好会 ...

  4. Hihocoder 1325 平衡树·Treap(平衡树,Treap)

    Hihocoder 1325 平衡树·Treap(平衡树,Treap) Description 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? 小Ho:就是二叉 ...

  5. HihoCoder 1325 平衡树·Treap

    HihoCoder 1325 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说 ...

  6. Hihocoder 1329 平衡树·Splay(平衡树)

    Hihocoder 1329 平衡树·Splay(平衡树) Description 小Ho:小Hi,上一次你跟我讲了Treap,我也实现了.但是我遇到了一个关键的问题. 小Hi:怎么了? 小Ho:小H ...

  7. Hihocoder 1333 (splay)

    Problem 平衡树 splay2 题目大意 维护一个序列,支持四种操作: 操作1:添加一个数,编号为x,权值为y. 操作2:删除编号在区间[x,y]内的数. 操作3:将编号在区间[x,y]内的数的 ...

  8. [模版]平衡树splay2

    题目描述 1. 加入:一个新的成员加入同好会,我会分配给他一个没有使用的id,并且询问他的兴趣值val. 2. 修改:id在区间[a,b]内的成员,兴趣值同时改变k,k有可能是负数,表示他们失去了对同 ...

  9. hihocoder #1034 : 毁灭者问题 平衡树(set)+线段树

    #1034 : 毁灭者问题 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 在 Warcraft III 之冰封王座中,毁灭者是不死族打三本后期时的一个魔法飞行单位. 毁 ...

随机推荐

  1. 使用Django框架实现游戏站点搭建

    完整project链接点击打开链接 上一篇中我们使用了Javascript和Html5实现了弹球游戏.而在本文中我们希望以其为基础实现游戏站点,可以实现用户的注冊登录.游戏成绩记录,排名显示.微博分享 ...

  2. Spring源代码解析和配置文件载入

    Spring类的继承结构图: Spring运用了大量的模板方法模式和策略模式,所以各位看源代码的时候,务必留意,每个继承的层次都有不同的作用.然后将同样的地方抽取出来,依赖抽象将不同的处理依照不同的策 ...

  3. Unix网络编程 高级IO套接字设置超时

    我们知道.对于一个套接字的读写(read/write)操作默认是堵塞的.假设当前套接字还不可读/写,那么这个操作会一直堵塞下去,这样对于一个须要高性能的server来说,是不能接受的.所以,我们能够在 ...

  4. MySQL 相邻两条数据相减

    <!-- 计算每两次消费的间隔天数 --> SELECT B.MEN_ID,TIMESTAMPDIFF(DAY,B.PRE_DATE,B.CURR_DATE) AS DAYS FROM ( ...

  5. Vim 复制粘贴探秘

    Vim作为最好用的文本编辑器之一,使用vim来编文档,写代码实在是很惬意的事情.每当学会了vim的一个新功能,就会很大地提高工作效率.有人使用vim几十年,还没有完全掌握vim的功能,这也说明了vim ...

  6. android 添加一个按键键值【转】

    本文转载自:http://blog.csdn.net/u012719256/article/details/52526046 1.frameworks/base/data/keyboards/Gene ...

  7. Reactive Native开发环境搭建

    root@zhongzhenhua-virtual-machine:~/AndroidCode# repo init -u https://android.googlesource.com/platf ...

  8. DIV+CSS设计时浏览器兼容性

          近期用Div+css做了个企业网站,在浏览器中测试的时候确发现在IE7中显示正常的页面,在ie6中非常混乱,当时第一感觉就想到了兼容问题,可是百思不得其解应该从哪下手,经过一两天的查资料, ...

  9. spark作业运行过程之--DAGScheduler

    DAGScheduler--stage划分和创建以及stage的提交 本篇,我会从一次spark作业的运行为切入点,将spark运行过程中涉及到的各个步骤,包括DAG图的划分,任务集的创建,资源分配, ...

  10. 每日算法——新型在线LCA

    在线LCA一般大家都会用倍增吧,时间复杂度O(nlogn),空间复杂度O(nlogn),都是非常严格的复杂度限制,并且各种边界处理比较麻烦,有没有更快更好的办法呢? 我们发现,在树链剖分时,我们不经意 ...