luogu P4115 Qtree4
题目链接
题解
动态点分治,和上一题一样.同样三个堆.就是带权,用边权替换深度就好
为什么要单独写这个题解呢,因为我卡常卡了一天....据说树剖比rmq快? 在第24次AC
同样也有更有做法
代码
#include<queue>
#include<cstdio>
#include<cctype>
#include<algorithm>
/*char ch;
char buf[100000],*p1 = buf,*p2 = buf;
int F = 1;
#define nc() \
p1 == p2 && (p2 = (p1 = buf) + fread(buf,1,100000,stdin),p1 == p2) ? EOF :*p1 ++;
#define read(x) \
x = 0;ch = nc();F = 1; \
while(!isdigit(ch)) { if (ch == '-') F = -1; ch = nc();}\
while(isdigit(ch))x = x*10+ch - '0',ch = nc();\
x *= F;
*/
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int MAXIN=3e5;
char IN[MAXIN],*SS=IN,*TT=IN;
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
inline int read() {
int now=0,f=1;register char c=gc();
for(;!isdigit(c);c=gc()) if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now*f;
}
#define INF 998244353
const int maxn = 100007;
int son[maxn],f[maxn],mn[maxn << 1][25],root,tot;
struct node {
int u,v,next,w;
} edge[maxn << 1];
int head[maxn],num = 0;
inline void add_edge(int u,int v,int w) {
edge[++ num].v = v;edge[num].next = head[u];head[u] = num;
edge[num].w = w;
}
int n;
int lg[maxn << 1],dfn;// = 0;
bool col[maxn];
struct Heap {
std::priority_queue<int>A,B;
inline void push(int x) { A.push(x); }
inline void erase(int x) { B.push(x); }
inline void pop() { while(B.size() && A.top() == B.top()) A.pop(),B.pop(); A.pop(); }
inline int top() {
while(B.size() && A.top() == B.top()) A.pop(),B.pop();
return A.top();
}
inline int size() { return A.size() - B.size(); }
int retop() { if(size() < 2) return 0;int x = top();pop();int ret = top();push(x);return ret;}
} b[maxn],c[maxn],ans;
inline void insert(Heap &s) {if(s.size() > 1)ans.push(s.top() + s.retop());}
inline void dele(Heap &s) {if(s.size() > 1) ans.erase(s.top() + s.retop());}
int pos[maxn],Dis[maxn];
int id[maxn],tm;
void dfs_rmq (int x,int fa) {
int t = ++ tm; mn[pos[x] = ++ dfn][0] = tm,id[t] = x;
for(int i = head[x];i;i = edge[i].next) {
int v = edge[i].v;
if(v == fa)continue;
Dis[v] = Dis[x] + edge[i].w;
dfs_rmq(v,x);
mn[++ dfn][0] = t; //访问完子树后加上Qwq
}
}
inline int lca(int x,int y) {
x = pos[x];y = pos[y];
if(y < x) std::swap(x,y);
int k = lg[y - x + 1];
return id[std::min(mn[x][k],mn[y - (1 << k) + 1][k])];
}
inline int dis(int x,int y) {
return Dis[x] + Dis[y] - 2 * Dis[lca(x,y)];
}
bool vis[maxn];int fa[maxn];
void get_root(int x,int fa) {
son[x] = 1; f[x] = 0;
for(int i = head[x];i;i = edge[i].next) {
int v = edge[i].v;
if(v == fa || vis[v]) continue;
get_root(v,x);
son[x] += son[v];f[x] = std::max(f[x],son[v]);
}
f[x] = std::max(f[x],tot - son[x]);
if(f[x] < f[root]) root = x;
}
void get(int x,int Fa,int rt) {
b[root].push(dis(x,fa[root]));
for(int i = head[x];i;i = edge[i].next) {
int v = edge[i].v; if(v == Fa || vis[v]) continue;
get(v,x,rt);
}
}
void build(int x,int Fa) {
fa[x] = Fa;vis[x] = 1;
c[x].push(0);
get(x,0,Fa);
for(int i = head[x];i;i = edge[i].next) {
int v = edge[i].v;
if(!vis[v] && v != Fa) {
tot = son[edge[i].v];root = 0;f[0] = INF;
get_root(edge[i].v,x);
v = root;
build(root,x);
c[x].push(b[v].top());
}
}
insert(c[x]);
}
void turn(int x,bool type) {
dele(c[x]);
if(type) c[x].erase(0);
else c[x].push(0);
insert(c[x]);
for(int i = x;i;i = fa[i]) {
dele(c[fa[i]]);
//printf("%d ",i);
if(b[i].size()) c[fa[i]].erase(b[i].top());
if(type)b[i].erase(dis(x,fa[i])); else b[i].push(dis(x,fa[i]));
if(b[i].size()) c[fa[i]].push(b[i].top());
insert(c[fa[i]]);
}
//puts("");
}
inline char get(){
char c=gc();
while(c!='A'&&c!='C') c=gc();
return c;
}
int main() {
n = read();
//read(n);
for(int u,v,w,i = 1;i < n;++ i) {
u = read(),v = read(),w = read();
//read(u);read(v);read(w);
add_edge(u,v,w);add_edge(v,u,w);
}
dfs_rmq(1,0);
for(int i = 2;i <= dfn;++ i) lg[i] = lg[i >> 1] + 1;
for(int i = 1;i <= lg[dfn];++ i)
for(int j = dfn - (1 << i - 1);j;-- j)
mn[j][i] = std::min(mn[j][i - 1],mn[j + (1 << i - 1)][i - 1]);
f[0] = INF; root = 0; tot = n;
get_root(1,0);
build(root,0);
//char opt[10];
int Q = read();
int sum = n;
for(int asd;Q --;) {
//scanf("%s",opt + 1);
if(get() == 'C') {
asd = read();//read(asd);
if(col[asd]) turn(asd,0), sum ++,col[asd] = 0;
else turn(asd,1),sum --,col[asd] = 1;
}
else {
if(sum == 1)puts("0");
else if(!sum)puts("They have disappeared.");
else printf("%d\n",std::max(0,ans.top()));
}
}
return 0;
}
luogu P4115 Qtree4的更多相关文章
- 洛谷 P2056 [ZJOI2007]捉迷藏 || bzoj 1095: [ZJOI2007]Hide 捉迷藏 || 洛谷 P4115 Qtree4 || SP2666 QTREE4 - Query on a tree IV
意识到一点:在进行点分治时,每一个点都会作为某一级重心出现,且任意一点只作为重心恰好一次.因此原树上任意一个节点都会出现在点分树上,且是恰好一次 https://www.cnblogs.com/zzq ...
- 洛谷 4115 Qtree4——链分治
题目:https://www.luogu.org/problemnew/show/P4115 论文:https://wenku.baidu.com/view/1bc2e4ea172ded630b1cb ...
- Luogu 魔法学院杯-第二弹(萌新的第一法blog)
虽然有点久远 还是放一下吧. 传送门:https://www.luogu.org/contest/show?tid=754 第一题 沉迷游戏,伤感情 #include <queue> ...
- luogu p1268 树的重量——构造,真正考验编程能力
题目链接:http://www.luogu.org/problem/show?pid=1268#sub -------- 这道题费了我不少心思= =其实思路和标称毫无差别,但是由于不习惯ACM风格的题 ...
- [luogu P2170] 选学霸(并查集+dp)
题目传送门:https://www.luogu.org/problem/show?pid=2170 题目描述 老师想从N名学生中选M人当学霸,但有K对人实力相当,如果实力相当的人中,一部分被选上,另一 ...
- [luogu P2647] 最大收益(贪心+dp)
题目传送门:https://www.luogu.org/problem/show?pid=2647 题目描述 现在你面前有n个物品,编号分别为1,2,3,--,n.你可以在这当中任意选择任意多个物品. ...
- Luogu 考前模拟Round. 1
A.情书 题目:http://www.luogu.org/problem/show?pid=2264 赛中:sb题,直接暴力匹配就行了,注意一下读入和最后一句话的分句 赛后:卧槽 怎么只有40 B.小 ...
- luogu P2580 于是他错误的点名开始了
luogu P2580 于是他错误的点名开始了 https://www.luogu.org/problem/show?pid=2580 题目背景 XS中学化学竞赛组教练是一个酷爱炉石的人. 他会一边 ...
- CJOJ 1331 【HNOI2011】数学作业 / Luogu 3216 【HNOI2011】数学作业 / HYSBZ 2326 数学作业(递推,矩阵)
CJOJ 1331 [HNOI2011]数学作业 / Luogu 3216 [HNOI2011]数学作业 / HYSBZ 2326 数学作业(递推,矩阵) Description 小 C 数学成绩优异 ...
随机推荐
- 【BZOJ】1875: [SDOI2009]HH去散步 矩阵快速幂
[题意]给定n个点m边的无向图,求A到B恰好经过t条边的路径数,路径须满足每条边都和前一条边不同.n<=20,m<=60,t<=2^30. [算法]矩阵快速幂 [题解]将图的邻接矩阵 ...
- 【BZOJ】4358: permu 莫队算法
[题意]给定长度为n的排列,m次询问区间[L,R]的最长连续值域.n<=50000. [算法]莫队算法 [题解]考虑莫队维护增加一个数的信息:设up[x]表示数值x往上延伸的最大长度,down[ ...
- JQuery的几个基础操作
先介绍两个函数(数组) 1.$.map(array,function(element,index)); 对于数组array中的每个元素,调用上面所示的function(element,index)函数 ...
- Ajax+innerHTML+Dgls=好的用户体验+高性能+高效率
为了引入Dgls,我们从创建Dom节点说起. 用JS创建Dom节点 var div = document.createElement('div'); div.className = 'gdls'; v ...
- C++类型转换 -- 由其他类型转换到自定义类型
由其他类型转换到自定义类型 由其他类型(如int,double)向自定义类的转换是由构造函数来实现,只有当类的定义和实现中提供了合适的构造函数,转换才能通过. /******************* ...
- 对RSA的认识
#没有经过专业老师的指导,所以您在阅读本文时建议参考即可. 学习视屏:https://www.youtube.com/watch?v=TqX0AHHwRYQ https://www.youtub ...
- Linux线程编程之生产者消费者问题【转】
转自:http://www.cnblogs.com/clover-toeic/p/4029269.html 前言 本文基于顺序循环队列,给出Linux生产者/消费者问题的多线程示例,并讨论编程时需要注 ...
- MySQL5.6 新特性之GTID【转】
转自 MySQL5.6 新特性之GTID - jyzhou - 博客园http://www.cnblogs.com/zhoujinyi/p/4717951.html 背景: MySQL5.6在5.5的 ...
- C基础 time.h 简单思路扩展
前言 - time 简单需求 时间业务相关代码. 基本属于框架的最底层. 涉及的变动都很小. 以前参与游戏研发时候, 这方面需求不少, 各种被策划花式吊打. 转行开发互联网服务之后很少遇到这方面需求. ...
- 《跟老齐学Python Django实战》读后感
1.说一下这本书,讲解的很细致,内容选取足够入门Django. 2.在学习这本书要注意的几点: <1>如果你想跟着敲这本书的代码必须要安装:Django版本1.10.1(当然也可以玩玩新版 ...