LCT模板
之前一直用的LCT模板,因为其实个人对LCT和Splay不是很熟,所以用起来总觉得略略的坑爹,过了一段时间就忘了,但事实上很多裸的LCT要改的东西是不多的,所以今天写了些注释,以后可能套起模板来会得心应手一点吧- -0
#pragma warning(disable:4996)
#include <iostream>
#include <cstring>
#include <string>
#include <cstdio>
#include <vector>
#include <algorithm>
#include <map>
using namespace std; #define ll long long
#define maxn 200000
#define INF 0x3f3f3f3f
#define INP INF+1
#define MP make_pair void merge(int mxx[2], int nxx[2], int pval, int pnum)
{
if (pval>mxx[0]){
mxx[1] = mxx[0]; nxx[1] = nxx[0];
mxx[0] = pval; nxx[0] = pnum;
}
else if (pval == mxx[0]){
nxx[0] += pnum;
}
else if (pval>mxx[1]){
mxx[1] = pval; nxx[1] = pnum;
}
else if(pval==mxx[1]){
nxx[1] += pnum;
}
} struct Node
{
/* Splay items(need not to be modified)*/
Node *p, *ch[2];
/* end */ /* LCT items(need not to be modified)*/
bool isRoot;
Node *fa;
/* end */ /* tags to add(rev should always exist)*/
bool rev;
int add, cov;
/* end */ /* data need to be maintained*/
int val, size;
int mx[2], num[2];
/* end */ /* default value for the sentinel null(all the TAG and DATA should be set correcly)*/
Node(){
/* DATA */
val = 0; size=0;
mx[0] = mx[1] = -INF;
num[0] = num[1] = 0;
/* TAG */
rev = 0;add = 0; cov = INP;
}
/* end */ /* Splay function(need not to be modified)*/
void setc(Node *c, int d){
ch[d] = c;
c->p = this;
}
bool d(){
return p->ch[1] == this;
}
/* end */ /* update function(maintain for ALL the DATA)*/
void upd(){
size = ch[0]->size + ch[1]->size + 1;
mx[0] = ch[0]->mx[0]; num[0] = ch[0]->num[0];
mx[1] = ch[0]->mx[1]; num[1] = ch[0]->num[1];
merge(mx, num, val, 1);
merge(mx, num, ch[1]->mx[0], ch[1]->num[0]);
merge(mx, num, ch[1]->mx[1], ch[1]->num[1]);
}
/* end */ /* tags function(for ALL the TAG,write a function)*/
void revIt(){
rev ^= 1;
swap(ch[0], ch[1]);
}
void addIt(int vx){
add += vx;
val += vx;
if (mx[0] != -INF) mx[0] += vx;
if (mx[1] != -INF) mx[1] += vx;
}
void setIt(int vx){
add = 0;
cov = vx;
val = vx;
mx[0] = vx; num[0] = size;
mx[1] = -INF; num[1] = 0;
}
/* end */ void relax();
void setRoot(Node *f);
}Tnull, *null = &Tnull; /* function setRoot(need not to be modified) */
void Node::setRoot(Node *f){
fa = f;
isRoot = true;
p = null;
} /* function relax(need to be rewrite for ALL the TAG) */
void Node::relax(){
if (cov != INP){
for (int i = 0; i<2; ++i){
if (ch[i] != null) ch[i]->setIt(cov);
}
cov = INP;
}
if (add != 0){
for (int i = 0; i < 2; i++){
if (ch[i] != null) ch[i]->addIt(add);
}
add = 0;
}
if (rev){
for (int i = 0; i < 2; i++){
if (ch[i] != null) ch[i]->revIt();
}
rev = 0;
}
} /* memory part(add C=mem for every iteration)*/
Node mem[maxn], *C = mem; /* function make(need to set the default value for ALL the TAG and DATA)*/
Node *make(int v){
/* DATA */
C->size=1;
C->val = v;
C->mx[0] = v; C->num[0] = 1;
C->mx[1] = -INF; C->num[1] = 0;
/* TAG */
C->rev = 0; C->add = 0;
C->cov = INP;
/* Pointer(need not to be modified) */
C->ch[0] = C->ch[1] = null; C->isRoot = true;
C->p = null;
C->fa = null;
return C++;
} /* function rot(need not to be modified) */
void rot(Node *t){
Node *p = t->p;
p->relax();
t->relax();
bool d = t->d();
p->p->setc(t, p->d());
p->setc(t->ch[!d], d);
t->setc(p, !d);
p->upd();
if (p->isRoot){
p->isRoot = false;
t->isRoot = true;
t->fa = p->fa;
}
} /* function pushTo(need not to be modified) */
void pushTo(Node*t) {
static Node*stk[maxn]; int top = 0;
while (t != null) {
stk[top++] = t;
t = t->p;
}
for (int i = top - 1; i >= 0; --i) stk[i]->relax();
} /* function splay(need not to be modified) */
void splay(Node*u, Node*f = null) {
pushTo(u);
while (u->p != f) {
if (u->p->p == f)
rot(u);
else
u->d() == u->p->d() ? (rot(u->p), rot(u)) : (rot(u), rot(u));
}
u->upd();
} /* Node pointer array*/
Node *v[maxn];
/* The graph stored in the E*/
vector<int> E[maxn];
int n, nQ; /* BFS items, using for construct the Tree */
int que[maxn], fa[maxn], qh = 0, qt = 0;
int wht[maxn]; /* function BFS(need not to be modified) */
void bfs()
{
qh = qt = 0;
que[qt++] = 1;
fa[1] = -1;
while (qh < qt){
int u = que[qh++];
for (int i = 0; i < E[u].size(); i++){
int e = E[u][i];
if (e != fa[u]){
fa[e] = u;
v[e]->fa = v[u];
que[qt++] = e;
}
}
}
} /* function expose(need not to be modified) */
Node *expose(Node *u)
{
Node *v;
for (v = null; u != null; v = u, u = u->fa){
splay(u);
u->ch[1]->setRoot(u);
u->setc(v, 1);
v->fa = u;
}
return v;
} /* function makeRoot(need not to be modified) */
void makeRoot(Node *u)
{
expose(u);
splay(u);
u->revIt();
} /* function addEdge(need not to be modified) */
void addEdge(Node *u, Node *v)
{
makeRoot(v);
v->fa = u;
} /* function delEdge(need not to be modified) */
void delEdge(Node *u, Node *v)
{
makeRoot(u);
expose(v); splay(u); u->setc(null, 1); u->upd();
v->setRoot(null);
} /* function find root(need not to be modified) */
Node *find_root(Node *u)
{
expose(u); splay(u);
while (u->ch[0] != null){
u = u->ch[0];
}
splay(u); return u;
} /* function used for query*/
/***************************
prototype:
void xPath(Node *u,Node *v,(int ax))
{
makeRoot(u);
expose(v);
splay(v);
v->xIt((ax))
}
***************************/
void addPath(Node *u, Node *v, int ax)
{
makeRoot(u);
expose(v);
splay(v);
v->addIt(ax);
} void setPath(Node *u, Node *v, int ax)
{
makeRoot(u);
expose(v);
splay(v);
v->setIt(ax);
} pair<int, int> queryPath(Node *u, Node *v){
makeRoot(u);
expose(v);
splay(v);
return MP(v->mx[1], v->num[1]);
} int main()
{
int T; cin >> T;int ca=0;
while (T--)
{
/* Things to do */
/*************************
C=mem;
initialize E;
read in the wht for nodes;
make for each node;
bfs();
solve for query();
**************************/
C = mem;
scanf("%d%d", &n, &nQ);
for (int i = 1; i <= n; i++){
scanf("%d", wht + i);
v[i] = make(wht[i]);
}
for (int i = 0; i <= n; i++) E[i].clear();
int ui, vi;
for (int i = 0; i < n - 1; i++){
scanf("%d%d", &ui, &vi);
E[ui].push_back(vi); E[vi].push_back(ui);
}
bfs();
int cmd, xi, yi, ai, bi;
Node *nx, *ny, *na, *nb;
printf("Case #%d:\n",++ca);
while (nQ--){
scanf("%d", &cmd);
if (cmd == 1) scanf("%d%d%d%d", &xi, &yi, &ai, &bi);
else if (cmd == 2 || cmd == 3) scanf("%d%d%d", &ai, &bi, &xi);
else scanf("%d%d", &ai, &bi); if (cmd == 1){
nx = v[xi]; ny = v[yi];
na = v[ai]; nb = v[bi];
delEdge(nx, ny);
addEdge(na, nb);
}
else if (cmd == 2){
nx = v[ai]; ny = v[bi];
setPath(nx, ny, xi);
}
else if (cmd == 3){
nx = v[ai]; ny = v[bi];
addPath(nx, ny, xi);
}
else{
nx = v[ai]; ny = v[bi];
pair<int, int> ans = queryPath(nx, ny);
if (ans.first <= -INF) printf("ALL SAME\n");
else printf("%d %d\n", ans.first, ans.second);
}
}
}
return 0;
}
LCT模板的更多相关文章
- LCT 模板及套路总结
这一个月貌似已经考了无数次\(LCT\)了..... 保险起见还是来一发总结吧..... A. LCT 模板 \(LCT\) 是由大名鼎鼎的 \(Tarjan\) 老爷发明的. 主要是用来维护树上路径 ...
- [洛谷P1501] [国家集训队]Tree II(LCT模板)
传送门 这是一道LCT的板子题,说白了就是在LCT上支持线段树2的操作. 所以我只是来存一个板子,并不会讲什么(再说我也不会,只能误人子弟2333). 不过代码里的注释可以参考一下. Code #in ...
- LuoguP3690 【模板】Link Cut Tree (动态树) LCT模板
P3690 [模板]Link Cut Tree (动态树) 题目背景 动态树 题目描述 给定n个点以及每个点的权值,要你处理接下来的m个操作.操作有4种.操作从0到3编号.点从1到n编号. 0:后接两 ...
- BZOJ2002 & LCT模板(分块不会搞)
题意: 看题. 某天,Lostmonkey发明了一种超级弹力装置,为了在他的绵羊朋友面前显摆,他邀请小绵羊一起玩个游戏.游戏一开始,Lostmonkey在地上沿 着一条直线摆上n个装置,每个装置设定初 ...
- bzoj2049-洞穴勘测(动态树lct模板题)
Description 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好 ...
- Luogu 3690 LCT - 模板
推荐几篇比较好的博客: FlashHu 的 讲解比较好 : 传送门 Candy 的 代码~ : 传送门 以及神犇Angel_Kitty的 学习笔记: 传送门 Code V 模板 #include< ...
- BZOJ 1180 / 2843 LCT模板题_双倍经验
一大早上到机房想先拍一下模板,热热身. 结果....对照着染色敲的 LCT 竟然死活也调不过去(你说我抄都能抄错) 干脆自己重新敲了一遍,10min就敲完了....... 还是要相信自己 Code: ...
- BZOJ3282: Tree (LCT模板)
Description 给定N个点以及每个点的权值,要你处理接下来的M个操作. 操作有4种.操作从0到3编号.点从1到N编号. 0:后接两个整数(x,y),代表询问从x到y的路径上的点的权值的xor和 ...
- LCT模板(学习笔记)(洛谷3690)(加边,删边,修改点权)
最近学习了一波LCT qwq 强势安利Flashhu的博客!!!!! 真的特别详细(可惜我不会弄链接) 如果有想要学习\(LCT\)的同学,可以直接看他的博客 我这里就简单写一点自己的体会啊. \(L ...
随机推荐
- Windows上搭建android开发环境
在搭建android开发环境时需要四部分内容,框架如下 其中Java SDK和Eclipse在java4android中有过介绍,重点介绍ADT和Android SDK的安装. 安装Android S ...
- opengl基础学习专题 (二) 点直线和多边形
题外话 随着学习的增长,越来越觉得自己很水.关于上一篇博文中推荐用一个 学习opengl的 基于VS2015的 simplec框架.存在 一些问题. 1.这个框架基于VS 的Debug 模式下,没有考 ...
- iOS学习之C语言数据类型
1.进制 0 1 2 3 4 5 6 7 8 9 A B C D E F 0X123(十六进制) 0123(八进制) 位权:单位数字的基本数值 2.数据类型 关键字 说明 字节大小 char ...
- HTML/CSS的学习过程一览
HTML/CSS的学习过程一览 说明 调试工具使用的是Google Chrome浏览器,其余浏览器出现的问题,这锅我不背[傲娇脸 可以使用浏览器查看源代码 网页列表 HTML_CSS_1 HTML基本 ...
- iOS 进阶 第十七天(0420)
0420 凡是继承了UIResponder的类都可以做响应者 响应事件的传递是由底到高来传递,响应者链条是由高到底来响应 相应事件的传递(由底到高 找到正在和用户触摸交互的view) 准则:事件由父控 ...
- To add private variable to this Javascript literal object
You can use number as function/variable name, the numberic name can't be accessed from parent scope, ...
- c enum用法
c语言中enum的用法,先用关键字enum声明一个类型如enum HUMAN {MAN ,WOMAN};这里就声明了一个HUMAN这个枚举类型.其中MAN的值为0,后面依次递增.后面要用的时候,先声明 ...
- C中的一些函数
简述:printf.sprintf函数 转载自http://www.cnblogs.com/adslg/archive/2008/08/22/1274164.html 部分进行了修改,参考http:/ ...
- Linux查看端口使用状态及启动
LINUX网络性能之管理工具三剑客 本文是介绍管理Linux查看端口这些输出信息,该命令将显示从每个数据包传出的头和来自主机hostname对端口80的编址.Netstat -tln 命令是Linux ...
- 《梦断代码》读书笔记第0篇——“软件时间”、“死定了”、“Agenda之魂“
第0章 软件时间 在未读这本书前,刚看到名字觉得是本讲代码的书,后来老师说是一个个的故事,这引起了我的兴趣,于是我便速速开始了第0章的阅读,读完一遍大概能读懂在讲什么,可能由于是译过来的书,书里面一 ...