BZOJ 3639: Query on a tree VII
Description
一棵树,支持三种操作,修改点权,修改颜色,问所有与他路径上颜色相同的点的最大权,包含这两个点.
Sol
LCT.
用LCT来维护重边,对于每个节点在建一个set用来维护轻边,这样Link和Cut是时候就非常好操作了,直接Access一下,Splay一下,直接删掉就可以了.
因为set是不统计重边的,然后对于每个节点的信息由他的父亲来保存,因为一个节点可能有很多儿子但一定只有一个父亲.
还有一个问题就是每个点的权值不能建全局的,因为维护的两颗LCT不能够同时删除,所以每个LCT都要有个点权的数组.
Code
/**************************************************************
Problem: 3639
User: BeiYu
Language: C++
Result: Accepted
Time:4060 ms
Memory:16276 kb
****************************************************************/ #include <bits/stdc++.h>
using namespace std; #define debug(a) cout<<#a<<"="<<a<<" "
typedef long long LL;
const int N = 1e5+50; inline int in(int x=0,char ch=getchar(),int v=1) {
while(ch>'9' || ch<'0') v=ch=='-'?-1:v,ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();return x*v; } int n,q;
vector < int > g[N];
int col[N],F[N]; struct LinkCutTree {
int f[N],ch[N][2];
int mx[N],w[N];
multiset< int,greater< int > > s[N]; #define lc(o) ch[o][0]
#define rc(o) ch[o][1] int isrt(int o) { return f[o]==0 || (lc(f[o])!=o && rc(f[o])!=o); }
void Update(int o) {
mx[o]=w[o];
if(!s[o].empty()) mx[o]=max(mx[o],*s[o].begin());
if(lc(o)) mx[o]=max(mx[o],mx[lc(o)]);
if(rc(o)) mx[o]=max(mx[o],mx[rc(o)]);
}
void Rot(int o) {
int p=f[o],k=f[p],r=rc(p)==o;
if(!isrt(p)) ch[k][rc(k)==p]=o;
f[ch[o][r^1]]=p,f[p]=o,f[o]=k;
ch[p][r]=ch[o][r^1],ch[o][r^1]=p;
Update(p),Update(o);
}
void Splay(int o) {
// cout<<"S"<<endl;
for(;!isrt(o);) {
int p=f[o],k=f[p];
if(isrt(p)) Rot(o);
else if((rc(p)==o)==(rc(k)==p)) Rot(p),Rot(o);
else Rot(o),Rot(o);
}Update(o);
}
void Access(int o) {
// cout<<"A"<<endl;
for(int p=0;o;p=o,o=f[o]) {
Splay(o);
if(rc(o)) s[o].insert(mx[rc(o)]);
if(rc(o)=p,p) s[o].erase(s[o].find(mx[p]));
}
}
void Link(int o) {
// cout<<"L"<<endl;
Access(F[o]),Splay(F[o]),Splay(o);
f[o]=F[o],rc(f[o])=o;
}
void Cut(int o) {
// cout<<"C"<<endl;
Access(o),Splay(o),f[lc(o)]=0,lc(o)=0;
}
void Change(int o,int v) {
// cout<<"M"<<endl;
Access(o),Splay(o),w[o]=v,Update(o);
}
int Query(int o) {
// cout<<"Q"<<endl;
Access(o),Splay(o);
int x=o;
for(;lc(x);x=lc(x));Splay(x);
if(col[x]!=col[o]) return mx[rc(x)];
else return mx[x];
}
}py[2]; void AddEdge(int u,int v) { g[u].push_back(v); }
void DFS(int u,int fa) {
for(int i=0,v;i<(int)g[u].size();i++) if((v=g[u][i])!=fa) {
F[v]=u,py[col[v]].f[v]=u;
DFS(v,u);
// debug(py[col[v]].mx[v])<<endl;
py[col[v]].s[u].insert(py[col[v]].mx[v]);
}py[0].Update(u),py[1].Update(u);
}
void init() {
n=in();
for(int i=1,u,v;i<n;i++) {
u=in(),v=in(),AddEdge(u,v),AddEdge(v,u);
}
for(int i=1;i<=n;i++) col[i]=in();
for(int i=1;i<=n;i++) py[0].mx[i]=py[0].w[i]=py[1].mx[i]=py[1].w[i]=in();
DFS(1,1);
// debug(py[1].mx[1])<<endl;
} int main() {
// freopen("in.in","r",stdin);
init();
for(int q=in();q--;) {
int opt=in(),u=in(),v;
if(opt==0) printf("%d\n",py[col[u]].Query(u));
else if(opt==1) {
if(F[u]) py[col[u]].Cut(u);
col[u]^=1;
if(F[u]) py[col[u]].Link(u);
} else {
v=in();
py[0].Change(u,v);
py[1].Change(u,v);
}
}
return 0;
}
BZOJ 3639: Query on a tree VII的更多相关文章
- BZOJ 3639: Query on a tree VII LCT_set维护子树信息
用 set 维护子树信息,细节较多. Code: #include <cstring> #include <cstdio> #include <algorithm> ...
- bzoj 3637: Query on a tree VI 树链剖分 && AC600
3637: Query on a tree VI Time Limit: 8 Sec Memory Limit: 1024 MBSubmit: 206 Solved: 38[Submit][Sta ...
- bzoj3639: Query on a tree VII
Description You are given a tree (an acyclic undirected connected graph) with n nodes. The tree node ...
- [BZOJ 3637]Query on a tree VI
偶然看见了这题,觉得自己 QTREE.COT 什么的都没有刷过的真是弱爆了…… 一道思路很巧妙的题,终于是在约大爷的耐心教导下会了,真是太感谢约大爷了. 这题显然是树链剖分,但是链上维护的东西很恶心. ...
- 2019.02.17 spoj Query on a tree VII(链分治)
传送门 跟QTREE6QTREE6QTREE6神似,改成了求连通块里的最大值. 于是我们对每条链开一个heapheapheap维护一下即可. MDMDMD终于1A1A1A链分治了. 代码: #incl ...
- SP16580 QTREE7 - Query on a tree VII
Description 一棵树,每个点初始有个点权和颜色(0/1) 0 u :询问所有u,v 路径上的最大点权,要满足u,v 路径上所有点的颜色都相同 1 u :反转u 的颜色 2 u w :把u 的 ...
- BZOJ 3637: Query on a tree VI LCT_维护子树信息_点权转边权_好题
非常喜欢这道题. 点权转边权,这样每次在切断一个点的所有儿子的时候只断掉一条边即可. Code: #include <cstring> #include <cstdio> #i ...
- [spojQTREE7]Query on a tree VII
即QTREE5和QTREE6组合,即将原本维护子树范围内点数改为维护子树范围内最小值即可,由于最小值没有可减性,因此需要使用set (虽然形式上与QTREE5类似,但QTREE5维护的信息更巧妙一些, ...
- BZOJ 1803 Query on a tree III
树上主席树. 我靠这是第k小吧..... #include<iostream> #include<cstdio> #include<cstring> #includ ...
随机推荐
- 高品质开源工具Chloe.ORM:支持存储过程与Oracle
扯淡 这是一款高质量的.NET C#数据库访问框架(ORM).查询接口借鉴 Linq.借助 lambda 表达式,可以完全用面向对象的方式就能轻松执行多表连接查询.分组查询.聚合查询.插入数据.批量删 ...
- [板子]倍增LCA
倍增LCA板子,没有压行,可读性应该还可以.转载请随意. #include <cstdio> #include <cstring> #include <algorithm ...
- JSP中编译指令include与动作指令include的区别
include指令是编译阶段的指令,即include所包含的文件的内容是编译的时候插入到JSP文件中,JSP引擎在判断JSP页面未被修改, 否则视为已被修改.由于被包含的文件是在编译时才插入的,因此如 ...
- 用vue.js学习es6(二):let和const使用
一.运行及关闭运行: 在上一节中我们用shift+右击在C:\vue\es6文件夹中打开命令行使用:npm run dev,打开了我们的vue界面. 如果要关闭则在命令行中按住ctrl+C则可以关闭. ...
- booting logo & booting animation
開機第一張圖片: 圖片位置: linux_repo/vendor/mediatek/proprietary/bootable/bootloader/lk/dev/logo 因為 project 選用 ...
- Docket学习--Docker入门
什么是Docker? Docker是一个开源的引擎,可以轻松的为任何应用创建一个轻量级的.可移植的.自给自足的容器.开发者在笔记本上编译测试通过的容器可以批量地在生产环境中部署,包括VMs(虚拟机). ...
- 浅析jquery ajax异步调用方法中不能给全局变量赋值的原因及解决方法(转载)
在调用一个jquery的ajax方法时我们有时会需要该方法返回一个值或者给某个全局变量赋值,可是我们发现程序执行完后并没有获取到我们想要的值,这时很有可能是因为你用的是ajax的异步调用async:t ...
- com.panie 项目开发随笔_前后端框架考虑(2016.12.8)
(一) 近日和一同学联系,说了我想要做一个网站的打算.她很感兴趣.于是我们协商了下,便觉得一起合作.她写前端,我写后台.因为我对于前端样式设计并不怎么熟悉. (二) 我们决定先做一个 个人博客. 网上 ...
- shell-for循环
sheel语言for循环格式 for var in item1 item2 ... itemN do command1 command2 ... commandN done 案例1 #!/bin/ba ...
- jQuery文件上传插件Uploadify(转)
一款基于flash的文件上传,有进度条和支持大文件上传,且可以多文件上传队列. 这款在flash的基础上增加了html5的支持,所以在移动端也可以使用. 由于官方提供的版本是flash免费,html5 ...