洛谷 P1501 [国家集训队]Tree II 解题报告
P1501 [国家集训队]Tree II
题目描述
一棵\(n\)个点的树,每个点的初始权值为\(1\)。对于这棵树有\(q\)个操作,每个操作为以下四种操作之一:
+ u v c:将\(u\)到\(v\)的路径上的点的权值都加上自然数\(c\);- u1 v1 u2 v2:将树中原有的边\((u_1,v_1)\)删除,加入一条新边\((u_2,v_2)\),保证操作完之后仍然是一棵树;* u v c:将\(u\)到\(v\)的路径上的点的权值都乘上自然数\(c\);/ u v:询问\(u\)到\(v\)的路径上的点的权值和,求出答案对于\(51061\)的余数。
输入输出格式
输入格式:
第一行两个整数\(n,q\)
接下来\(n-1\)行每行两个正整数\(u,v\),描述这棵树
接下来\(q\)行,每行描述一个操作
输出格式:
对于每个/对应的答案输出一行
说明
\(10\%\)的数据保证,\(1<=n\),\(q<=2000\)
另外\(15\%\)的数据保证,\(1<=n\),\(q<=5*10^4\),没有-操作,并且初始树为一条链
另外\(35\%\)的数据保证,\(1<=n\),\(q<=5*10^4\),没有-操作
\(100\%\)的数据保证,\(1<=n,q<=10^5\),\(0<=c<=10^4\)
就是个LCTsb题
然后我放乘法标记的时候忘记放到加法标记上了,记录一下自己的sb错误。
注意要开uint
Code:
#include <cstdio>
#define ls ch[now][0]
#define rs ch[now][1]
#define fa par[now]
const unsigned int mod=51061;
const unsigned int N=1e5+10;
unsigned int ch[N][2],par[N],sum[N],dat[N],tag[N],add[N],mul[N],siz[N],s[N],tot,tmp,n,m;
bool isroot(int now){return ch[fa][0]==now||ch[fa][1]==now;}
unsigned int identity(int now){return ch[fa][1]==now;}
void connect(int f,int now,int typ){ch[f][typ]=now,fa=f;}
void Reverse(int now){tag[now]^=1,tmp=ls,ls=rs,rs=tmp;}
void updata(int now){sum[now]=(sum[ls]+sum[rs]+dat[now])%mod;siz[now]=siz[ls]+siz[rs]+1;}
void pushdown(int now)
{
if(tag[now]){if(ls) Reverse(ls);if(rs) Reverse(rs);tag[now]^=1;}
if(ls)
{
sum[ls]=(sum[ls]*mul[now]+add[now]*siz[ls])%mod;
dat[ls]=(dat[ls]*mul[now]+add[now])%mod;
mul[ls]=mul[ls]*mul[now]%mod;
add[ls]=(add[ls]*mul[now]+add[now])%mod;
}
if(rs)
{
sum[rs]=(sum[rs]*mul[now]+add[now]*siz[rs])%mod;
dat[rs]=(dat[rs]*mul[now]+add[now])%mod;
mul[rs]=mul[rs]*mul[now]%mod;
add[rs]=(add[rs]*mul[now]+add[now])%mod;
}
add[now]=0,mul[now]=1;
}
void Rotate(int now)
{
int p=fa,typ=identity(now);
connect(p,ch[now][typ^1],typ);
if(isroot(p)) connect(par[p],now,identity(p));
else fa=par[p];
connect(now,p,typ^1);
updata(p),updata(now);
}
void splay(int now)
{
while(isroot(now)) s[++tot]=now,now=fa;
s[++tot]=now;
while(tot) pushdown(s[tot--]);
now=s[1];
for(;isroot(now);Rotate(now))
if(isroot(fa))
Rotate(identity(now)^identity(fa)?now:fa);
}
void access(int now)
{
for(int las=0;now;las=now,now=fa)
splay(now),rs=las,updata(now);
}
void evert(int now){access(now),splay(now),Reverse(now);}
void link(int u,int v){evert(u),par[u]=v;}
void makeline(int u,int v){evert(u),access(v),splay(v);}
void cat(int u,int v){makeline(u,v),ch[v][0]=par[u]=0,updata(v);}
unsigned int query(int u,int v){makeline(u,v);return sum[v];}
void Add(int u,int v,unsigned int d)
{
makeline(u,v);
(dat[v]+=d)%=mod,(sum[v]+=d*siz[v])%=mod,(add[v]+=d)%=mod;
}
void Mul(int u,int v,unsigned int d)
{
makeline(u,v);
(dat[v]*=d)%=mod,(sum[v]*=d)%=mod,(add[v]*=d)%=mod,(mul[v]*=d)%=mod;
}
int main()
{
scanf("%d%d",&n,&m);
for(unsigned int i=1;i<=n;i++) siz[i]=dat[i]=sum[i]=mul[i]=1;
for(unsigned int u,v,i=1;i<n;i++)
{
scanf("%d%d",&u,&v);
link(u,v);
}
char op[3];
for(unsigned int u,v,c,i=1;i<=m;i++)
{
scanf("%s%d%d",op,&u,&v);
if(op[0]=='+') scanf("%d",&c),Add(u,v,c);
else if(op[0]=='-') cat(u,v),scanf("%d%d",&u,&v),link(u,v);
else if(op[0]=='*') scanf("%d",&c),Mul(u,v,c);
else printf("%d\n",query(u,v));
}
return 0;
}
2018.12.7
洛谷 P1501 [国家集训队]Tree II 解题报告的更多相关文章
- 洛谷P1501 [国家集训队]Tree II(LCT,Splay)
洛谷题目传送门 关于LCT的其它问题可以参考一下我的LCT总结 一道LCT很好的练习放懒标记技巧的题目. 一开始看到又做加法又做乘法的时候我是有点mengbi的. 然后我想起了模板线段树2...... ...
- 【刷题】洛谷 P1501 [国家集训队]Tree II
题目描述 一棵n个点的树,每个点的初始权值为1.对于这棵树有q个操作,每个操作为以下四种操作之一: + u v c:将u到v的路径上的点的权值都加上自然数c: - u1 v1 u2 v2:将树中原有的 ...
- 洛谷P1501 [国家集训队]Tree II(LCT)
题目描述 一棵n个点的树,每个点的初始权值为1.对于这棵树有q个操作,每个操作为以下四种操作之一: + u v c:将u到v的路径上的点的权值都加上自然数c: - u1 v1 u2 v2:将树中原有的 ...
- 洛谷P1501 [国家集训队]Tree II(打标记lct)
题目描述 一棵n个点的树,每个点的初始权值为1.对于这棵树有q个操作,每个操作为以下四种操作之一: + u v c:将u到v的路径上的点的权值都加上自然数c: - u1 v1 u2 v2:将树中原有的 ...
- [洛谷P1501][国家集训队]Tree II
题目大意:给一棵树,有四种操作: $+\;u\;v\;c:$将路径$u->v$区间加$c$ $-\;u_1\;v_1\;u_2\;v_2:$将边$u_1-v_1$切断,改成边$u_2-v_2$, ...
- 洛谷 P1501 [国家集训队]Tree II
看来这个LCT板子并没有什么问题 #include<cstdio> #include<algorithm> using namespace std; typedef long ...
- 洛谷 P1501 [国家集训队]Tree II Link-Cut-Tree
Code: #include <cstdio> #include <algorithm> #include <cstring> #include <strin ...
- [洛谷P1501] [国家集训队]Tree II(LCT模板)
传送门 这是一道LCT的板子题,说白了就是在LCT上支持线段树2的操作. 所以我只是来存一个板子,并不会讲什么(再说我也不会,只能误人子弟2333). 不过代码里的注释可以参考一下. Code #in ...
- 洛谷 P1852 [国家集训队]跳跳棋 解题报告
P1852 [国家集训队]跳跳棋 题目描述 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在\(a\),\(b\), ...
随机推荐
- 原生js使用ajax
AJAX 可以在不重新加载整个页面的情况下,与服务器交换数据并更新部分网页内容(默认是异步) (1)使用ajax会用到XMLHttpRequest()对象 (2)然后使用open方法定义请求方法和请求 ...
- selenium +java 多个类公用driver问题
问题点:太久没有写selenium代码,居然把driver公用的问题忘记了,即:每写一个测试类,执行过程中都会新建一个窗口,这样应该说是非常不专业的. 大概想了一个方法,虽然看起来也不怎么专业,但感觉 ...
- json_encode替代函数
<?php function jsonEncode($var) { if (function_exists('json_encode')) { return json ...
- Linux 优化详解
一.引子 系统优化是一项复杂.繁琐.长期的工作,优化前需要监测.采集.测试.评估,优化后也需要测试.采集.评估.监测,而且是一个长期和持续的过程,不是说现在又花了.测试了,以后就可以一劳永逸,而不是说 ...
- 使用Photon引擎进行unity网络游戏开发(四)——Photon引擎实现网络游戏逻辑
使用Photon引擎进行unity网络游戏开发(四)--Photon引擎实现网络游戏逻辑 Photon PUN Unity 网络游戏开发 网络游戏逻辑处理与MasterClient 网络游戏逻辑处理: ...
- 【数据结构系列】线段树(Segment Tree)
一.线段树的定义 线段树,又名区间树,是一种二叉搜索树. 那么问题来了,啥是二叉搜索树呢? 对于一棵二叉树,若满足: ①它的左子树不空,则左子树上所有结点的值均小于它的根结点的值 ②若它的右子树不空, ...
- Zabbix部署-LNMP环境
原文发表于cu:2016-05-05 参考文档: LNMP安装:http://www.osyunwei.com/archives/7891.html 一.环境 Server:CentOS-7-x86_ ...
- cs231n学习笔记(一)计算机视觉及其发展史
在网易云课堂上学习计算机视觉经典课程cs231n,觉得有必要做个笔记,因为自己的记性比较差,留待以后查看. 每一堂课都对应一个学习笔记,下面就开始第一堂课. 这堂课主要是回顾了计算机视觉的起源及其后来 ...
- 数据挖掘学习笔记——kaggle 数据预处理
预处理 1. 删除缺失值 a. 删除行即样本(对于样本如果输出变量存在缺失的则直接删除该行,因为无法用该样本训练) b. 删除列,即特征(采用这种删除方式,应保证训练集和验证集都应当删除相同的特征) ...
- 电脑提示‘您需要来自Administration的权限才能对此文件夹进行更改’怎么删除文件
电脑提示'您需要来自Administration的权限才能对此文件夹进行更改'怎么删除文件 应该怎么做 win7系统需要定期删除一些无用的文件,扩大内存空间,但是在删除文件的时候弹出提示"您 ...