• Link-Cut-Tree 的懒标记下传正确食用方法。

我们来逐步分析每一个操作。


1:+ u v c:将u到v的路径上的点的权值都加上自然数c;

  • 解决方法:

  • 很显然,我们可以 split(u,v) 来提取u,v这一段区间,提取完了将 Splay(v),然后直接在v上打加法标记add即可。

  • 代码:

inline void pushadd(ll x,ll val){//打标记
s[x]+=sz[x]*val,v[x]+=val,add[x]+=val;
s[x]%=MOD,v[x]%=MOD,add[x]%=MOD;
} inline void split(ll x,ll y){//LCT基本操作split,不再赘述
makeroot(x);Access(y);Splay(y);
} //(main函数中):
if(op[0]=='+'){
scanf("%lld%lld%lld",&x,&y,&v);//输入信息
split(x,y);pushadd(y,v);//提取链条&打标记
}

2:- u1 v1 u2 v2:将树中原有的边(u1,v1)删除,加入一条新边(u2,v2),保证操作完之后仍然是一棵树;

  • 解决方法:

  • 删除边即cut操作,加边即link操作。

  • 代码:

inline void link(ll x,ll y){
makeroot(x);if(findroot(x)!=y)f[x]=y;
}
inline void cut(ll x,ll y){
makeroot(x);split(x,y);
if(findroot(y)==x&&f[x]==y&&!ch[x][1])
f[x]=ch[y][0]=0;return;
}//LCT基本操作link&cut,不再赘述 //(main函数中):
if(op[0]=='-'){
scanf("%lld%lld",&x,&y);cut(x,y);//删边
scanf("%lld%lld",&x,&y);link(x,y);//加边
}

3:* u v c:将u到v的路径上的点的权值都乘上自然数c;

  • 解决方法:

  • 很显然,我们可以split(u,v)来提取u,v这一段区间,提取完了将Splay(v),然后直接在v上打乘法标记mul即可。(跟第一个操作基本同理)

  • 代码:

inline void pushmul(ll x,ll val){//打标记
s[x]*=val,v[x]*=val,mul[x]*=val,add[x]*=val;
s[x]%=MOD,v[x]%=MOD,mul[x]%=MOD,add[x]%=MOD;
} //(main函数中):
if(op[0]=='*'){
scanf("%lld%lld%lld",&x,&y,&v);
split(x,y);pushmul(y,v);
}

4:/ u v:询问u到v的路径上的点的权值和,求出答案对于51061的余数。

  • 解决方法:

  • 我们可以像第1、3操作那样,先扯出这条链来(split(u,v)),因为split操作时最后Splay过了,根节点是v。而Splay的时候又将整棵Splay上的节点信息都跟新好了(懒标记都下传了),所以这棵Splay的根节点的点权即为这课Splay的点权和。而这课Splay又代表着u,v这一段区间,所以最后只需输出s[v]即可。

  • 代码:

//(main函数中):
if(op[0]=='/'){
scanf("%lld%lld",&x,&y);
split(x,y);printf("%lld\n",s[y]);
}

Code:

#include<bits/stdc++.h>
#define ll long long
#define inf 0x3f3f3f3f
#define RI register ll
#define A printf("A")
#define C printf(" ")
#define MOD 51061
using namespace std;
const ll N=1e5+2;
template<typename _Tp> inline void IN(_Tp& dig){
char c;bool flag=0;dig=0;
while(c=getchar(),!isdigit(c))if(c=='-')flag=1;
while(isdigit(c))dig=dig*10+c-'0',c=getchar();
if(flag)dig=-dig;
}ll f[N],s[N],v[N],sz[N],rev[N],mul[N],add[N],hep[N],ch[N][2];
inline ll get(ll x){return ch[f[x]][0]==x||ch[f[x]][1]==x;}
inline ll chk(ll x){return ch[f[x]][1]==x;}
inline void pushfilp(ll x){
swap(ch[x][0],ch[x][1]);rev[x]^=1;
}
inline void pushup(ll x){
s[x]=(s[ch[x][0]]+s[ch[x][1]]+v[x])%MOD;
sz[x]=sz[ch[x][0]]+sz[ch[x][1]]+1;
}
inline void pushmul(ll x,ll val){
s[x]*=val,v[x]*=val,mul[x]*=val,add[x]*=val;
s[x]%=MOD,v[x]%=MOD,mul[x]%=MOD,add[x]%=MOD;
}
inline void pushadd(ll x,ll val){
s[x]+=sz[x]*val,v[x]+=val,add[x]+=val;
s[x]%=MOD,v[x]%=MOD,add[x]%=MOD;
}
inline void pushdown(ll x){
if(mul[x]!=1)pushmul(ch[x][0],mul[x]),pushmul(ch[x][1],mul[x]);
if(add[x])pushadd(ch[x][0],add[x]),pushadd(ch[x][1],add[x]);
if(rev[x]){
if(ch[x][0])pushfilp(ch[x][0]);
if(ch[x][1])pushfilp(ch[x][1]);
}rev[x]=0,add[x]=0,mul[x]=1;return;
}
inline void rotate(ll x){
ll y=f[x],z=f[y],k=chk(x),v=ch[x][!k];
if(get(y))ch[z][chk(y)]=x;ch[x][!k]=y,ch[y][k]=v;
if(v)f[v]=y;f[y]=x,f[x]=z;pushup(y),pushup(x);
}
inline void Splay(ll x){
ll y=x,top=0;hep[++top]=y;
while(get(y))hep[++top]=y=f[y];
while(top)pushdown(hep[top--]);
while(get(x)){
y=f[x],top=f[y];
if(get(y))rotate((ch[y][0]==x)^(ch[top][0]==y)?y:x);
rotate(x);
}pushup(x);return;
}
inline void Access(ll x){
for(register ll y=0;x;x=f[y=x])
Splay(x),ch[x][1]=y,pushup(x);
}
inline ll findroot(ll x){
Access(x);Splay(x);
while(ch[x][0])pushdown(x),x=ch[x][0];
return x;
}
inline void makeroot(ll x){
Access(x);Splay(x);pushfilp(x);
}
inline void split(ll x,ll y){
makeroot(x);Access(y);Splay(y);
}
inline void link(ll x,ll y){
makeroot(x);if(findroot(x)!=y)f[x]=y;
}
inline void cut(ll x,ll y){
makeroot(x);split(x,y);
if(findroot(y)==x&&f[x]==y&&!ch[x][1])
f[x]=ch[y][0]=0;return;
}char op[2];
int main(){
ll n,m,x,y;scanf("%lld%lld",&n,&m);
for(register int i=1;i<=n;++i)
mul[i]=sz[i]=v[i]=1;ll v;
for(register int i=1;i<n;++i)
scanf("%lld%lld",&x,&y),link(x,y);
for(register int i=1;i<=m;++i){
scanf("%s",op);
if(op[0]=='+'){
scanf("%lld%lld%lld",&x,&y,&v);
split(x,y);pushadd(y,v);
}else if(op[0]=='-'){
scanf("%lld%lld",&x,&y);cut(x,y);
scanf("%lld%lld",&x,&y);link(x,y);
}else if(op[0]=='*'){
scanf("%lld%lld%lld",&x,&y,&v);
split(x,y);pushmul(y,v);
}else if(op[0]=='/'){
scanf("%lld%lld",&x,&y);
split(x,y);printf("%lld\n",s[y]);
}
}return 0;
}

因为51061 * 5106是会越过int界限的,所以我开的longlong(当然也可以开无符号int)

我居然因为没开longlong调了两个多小时.....

题解 洛谷P1501/BZOJ2631【[国家集训队]Tree II】的更多相关文章

  1. 【洛谷 P1501】 [国家集训队]Tree II(LCT)

    题目链接 Tree Ⅱ\(=\)[模板]LCT+[模板]线段树2.. 分别维护3个标记,乘的时候要把加法标记也乘上. 还有就是模数的平方刚好爆\(int\),所以开昂赛德\(int\)就可以了. 我把 ...

  2. 【洛谷1501】[国家集训队] Tree II(LCT维护懒惰标记)

    点此看题面 大致题意: 有一棵初始边权全为\(1\)的树,四种操作:将两点间路径边权都加上一个数,删一条边.加一条新边,将两点间路径边权都加上一个数,询问两点间路径权值和. 序列版 这道题有一个序列版 ...

  3. 洛谷 P1501 [国家集训队]Tree II 解题报告

    P1501 [国家集训队]Tree II 题目描述 一棵\(n\)个点的树,每个点的初始权值为\(1\).对于这棵树有\(q\)个操作,每个操作为以下四种操作之一: + u v c:将\(u\)到\( ...

  4. P1501 [国家集训队]Tree II(LCT)

    P1501 [国家集训队]Tree II 看着维护吧2333333 操作和维护区间加.乘线段树挺像的 进行修改操作时不要忘记吧每个点的点权$v[i]$也处理掉 还有就是$51061^2=2607225 ...

  5. BZOJ 2631 [国家集训队]Tree II (LCT)

    题目大意:给你一棵树,让你维护一个数据结构,支持 边的断,连 树链上所有点点权加上某个值 树链上所有点点权乘上某个值 求树链所有点点权和 (辣鸡bzoj又是土豪题,洛谷P1501传送门) LCT裸题, ...

  6. 洛谷P1501 [国家集训队]Tree II(LCT,Splay)

    洛谷题目传送门 关于LCT的其它问题可以参考一下我的LCT总结 一道LCT很好的练习放懒标记技巧的题目. 一开始看到又做加法又做乘法的时候我是有点mengbi的. 然后我想起了模板线段树2...... ...

  7. ⌈洛谷1505⌋⌈BZOJ2157⌋⌈国家集训队⌋旅游【树链剖分】

    题目链接 [洛谷] [BZOJ] 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T ...

  8. 树链剖分【洛谷P1505】 [国家集训队]旅游

    P1505 [国家集训队]旅游 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城 ...

  9. 【洛谷】P4555 [国家集训队]最长双回文串

    P4555 [国家集训队]最长双回文串 题源:https://www.luogu.com.cn/problem/P4555 原理:Manacher 还真比KMP好理解 解决最长回文串问题 转化为长度为 ...

随机推荐

  1. Swing中子元素截获MouseEvent问题

    在父元素中绑定MouseMotion监听,但是当鼠标在子元素中时父元素无法收到 这时候需要在子元素中绑定MouseMotion,然后使用: getParent().dispatchEvent(e); ...

  2. Cocos2d-x 3.2编译生成Android程序出错Error running command, return code: 2的解决方法

    用Cocos2d-x 3.2正式版创建项目,结果使用cocos compile -p android编译生成APK程序,结果悲剧了,出现以下错误. Android NDK: Invalid APP_S ...

  3. linux下的C语言开发(定时器)

    定时器是我们需要经常处理的一种资源.那Linux下面的定时器又是怎么一回事呢?其实,在linux里面有一种进程中信息传递的方法,那就是信号.这里的定时器就相当于系统每隔一段时间给进程发一个定时信号,我 ...

  4. FreeMarker:模板开发指南

    ylbtech-FreeMarker:模板开发指南 1.返回顶部 1. Section Contents 入门 模板 + 数据模型 = 输出 数据模型一览 模板一览 数值,类型 基本内容 类型 模板 ...

  5. mysql status关键字 数据表设计中慎重使用

    mysql status关键字  数据表设计中慎重使用

  6. LVS的持久连接、会话保持和高可用介绍

    持续连接 1)持久连接(lvs persistence)模板: 实现无论使用任何调度算法,在一段时间内(默认360s),能够实现将来自同一个地址的请求始终发往同一个RS ipvsadm -A|E -t ...

  7. 符号修饰与函数签名、extern “C”(转载)

    转自:http://www.cnblogs.com/monotone/archive/2012/11/16/2773772.html 参考资料: <程序员的自我修养>3.5.3以及3.5. ...

  8. 通过usb访问mtp设备(ubuntu12.04) (转载)

    转自:http://robert.penz.name/658/howto-access-mtp-devices-via-usb-on-ubuntu-12-04/ A friend asked me h ...

  9. render same axis

    // 当前渲染相机的参数    QGlobalCamera* curRenderCamera = _getWorld()->getMainCam();    const Matrix4& ...

  10. bzoj 1702: [Usaco2007 Mar]Gold Balanced Lineup 平衡的队列【hash】

    我%&&--&()&%????? 双模hashWA,unsigned long longAC,而且必须判断hash出来的数不能为0???? 我可能学了假的hash 这个 ...