题目描述

植物学家Dustar培养出了一棵神奇的树,这棵有根树有n个节点,每个节点上都有一个数字a[i],而且这棵树的根为r节点。
这棵树非常神奇,可以随意转换根的位置,上一秒钟它的根是x节点,下一秒钟它的根就变成了y节点,Dustar由于魔力不够,无法控制住这棵树,这棵树说:“只要你每次都能正确答出以x为根的子树里的点的权值和我就停下来。”,你能帮帮Dustar吗?

题目大意

给你一棵树根不定的树,每次让你查询\(x\)节点的子树权值和。

解法

树链剖分求换根树。
对于每一个\(u\),我们假设\(root\)是当前整棵树的根。

  • 如果\(u=root\),那么答案就是整个树的权值和
  • 如果\(Lca(u,root)≠u\),就说明u的子树没有变,答案还是在原来的u的答案。
  • 如果\(Lca(u,root)=u\),即\(u\)在原来的树中是root 的祖先。那么我们找到\(u\)到\(root\)路径上的第一个儿子。这个儿子对应的原树中的子树,就是现在\(u\)的子树的补集。

ac代码

#include<iostream>
#define N 200005
#define lson nod<<1
#define rson nod<<1|1
using namespace std;
struct edge{int to,nt;};
edge E[N<<1];
int H[N],v[N],sz[N],fa[N],top[N],dep[N],son[N],pos[N],tot,idx[N],out[N],tree[N<<2],val[N],add[N<<2];
int cnt,n,m,rt;
int r(){
    int w=0,x=0;char ch=0;
    while(!isdigit(ch))w|=ch=='-',ch=getchar();
    while(isdigit(ch))x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
    return w?-x:x;
}
void pushup(int nod){tree[nod]=tree[lson]+tree[rson];}
void pushdown(int l,int r,int nod){
    if(!add[nod]) return;
    int mid=l+r>>1;
    add[lson]+=add[nod];
    add[rson]+=add[nod];
    tree[lson]+=(mid-l+1)*add[nod],tree[rson]+=(r-mid)*add[nod];
    add[nod]=0;
}
void addedge(int u,int v){
    E[++cnt]=(edge){v,H[u]};H[u]=cnt;
    E[++cnt]=(edge){u,H[v]};H[v]=cnt;
}
void dfs1(int u){
    sz[u]=1;
    for(int i=H[u];i;i=E[i].nt){
        int v=E[i].to;
        if(v==fa[u]) continue;
        fa[v]=u,dep[v]=dep[u]+1;
        dfs1(v); sz[u]+=sz[v];
        if(sz[v]>sz[son[u]]) son[u]=v;
    }
}
int Lca(int u,int v){//求最近公共祖先
    while(top[u]!=top[v]) {
        if(dep[top[u]]<dep[top[v]]) swap(u,v);
        u=fa[top[u]];
    }
    return dep[u]>dep[v]?v:u;
}
void dfs2(int u,int tp) {//求重边,dfs序
    pos[u]=++tot,idx[tot]=u;top[u]=tp;
    if(!son[u]) return void(out[u]=tot);
    dfs2(son[u],tp);
    for(int i=H[u];i;i=E[i].nt){
        int v=E[i].to;
        if(v==fa[u]||v==son[u]) continue;
        dfs2(v,v);
    }
    out[u]=tot;
}
void build(int l,int r,int nod){//建线段树
    if(l==r){tree[nod]=val[idx[l]]; return;}
    int mid=l+r>>1;
    build(l,mid,lson); build(mid+1,r,rson);
    pushup(nod);
}
int query(int l,int r,int ql,int qr,int nod){
    if(ql<=l&&r<=qr)return tree[nod];
    pushdown(l,r,nod);
    int res=0,mid=l+r>>1;
    if(ql<=mid) res+=query(l,mid,ql,qr,lson);
    if(qr>mid) res+=query(mid+1,r,ql,qr,rson);
    pushup(nod);
    return res;
}
int find(int u,int v){//找到u和v上的第一个儿子
    while(top[u]!=top[v]) {
        if(dep[top[u]]<dep[top[v]]) swap(u,v);
        if(fa[top[u]]==v) return top[u];
        u=fa[top[u]];
    }
    if(dep[u]<dep[v]) swap(u,v);
    return son[v];
}
int subquery(int u){
    if(u==rt) return query(1,n,1,n,1);
    int lca=Lca(u,rt);
    if(lca!=u) return query(1,n,pos[u],out[u],1);
    int son=find(u,rt),res=0;
    res+=query(1,n,1,n,1);
    res-=query(1,n,pos[son],out[son],1);
    return res;
}
int main(){
    n=r(),m=r();
    for(int i=1;i<n;i++)addedge(r(),r());
    for(int i=1;i<=n;i++)val[i]=r();
    rt=r(); dfs1(rt); dfs2(rt,rt);
    build(1,n,1);
    while(m--){
        int opt=r();
        if(opt==1)rt=r();
        if(opt==2)printf("%d\n",subquery(r()));
    }
    return 0;
}

[hgoi#2019/2/16t4]transform的更多相关文章

  1. 「HGOI#2019.4.19省选模拟赛」赛后总结

    t1-Painting 这道题目比较简单,但是我比较弱就只是写了一个链表合并和区间DP. 别人的贪心吊打我的DP,嘤嘤嘤. #include <bits/stdc++.h> #define ...

  2. [hgoi#2019/3/21]NOIP&NOI赛后总结

    前言 今天做的是是2010年提高组和NOI的题目,做过几道原题,但是还是爆炸了,我真的太弱了. t1-乌龟棋 https://www.luogu.org/problemnew/show/P1541 这 ...

  3. [hgoi#2019/3/10]赛后总结

    关于本次hg模拟赛,题目来源于CF1110. t1-无意义运算符(meaning) 题目描述 最大公约数和位运算之间有共同点吗?是时候来研究一下了. 给定一个正整数a,请找到一个闭区间[1,a-1] ...

  4. [hgoi#2019/3/3]赛后总结

    T1--最长公共前缀(lcp) 定义两个字符串S,T 的最长公共前缀lcp(S,T)为最长的字符串R,满足R 既是S 的前缀又是T 的前缀. 给定一个字符串S,下标从1 开始,每次询问给出四个正整数a ...

  5. [hgoi#2019/2/16t3]psolve

    题目描述 Dustar有n道题目要做.他的月薪是m元. 由于题目是一流的难题,所以Dustar不得不找个人来帮(代)助(替)他写作业. 找人写作业不是免费的,但是他们能保证在一个月内做出任何题目.每做 ...

  6. [hgoi#2019/2/16t2]friend

    题目描述 在一个遥远的国度里有n个人,每个人手上写着4个互不相同的数. 这个国度比较奇怪,如果两个人至少有一个数字相同,则他们是一对朋友. 现在这n个人按序号从左到右排成了一排,每个人都想知道在他左边 ...

  7. [hgoi#2019/2/16t1]math

    题目描述 解法 我们稍微枚举一下前面几位,可以得到这样的规律. \[X_i=\frac{1}{2^{i+1}-1}\] \[Y_i=\frac{1}{2^{2^i}-1}\] 那么要使\(xm=yn\ ...

  8. [hgoi#2019/2/24]玄学考试

    感想 对于这次考试,真的不想说什么了,太玄学了!!! t1输出比标准输出长,这是什么操作???难道要关文件???但是交到oj上又A掉了.这是什么操作. t2还好,没有出什么意外...但是要吐槽一下出题 ...

  9. [hgoi#2019/2/18]比较水

    T1--调换纸牌(card) Alex有 n张纸牌,每张纸牌上都有一个值ai,Alex把这些纸牌排成一排,希望将纸牌按值从小到大的顺序排好.现在他把这个任务交给你,你只能进行一种操作:选中一张牌,然后 ...

随机推荐

  1. Rabbitmq-topic演示

    在direct演示里,我们的日志系统实现了可选择性的接收日志.但仍旧有一些限制:不能基于多种标准进路由.在一个完整的日志系统中,我们可能不仅要根据日志的严重级别来接收日志,可能需要基于日志的来源来进行 ...

  2. BootStrap学习(7)_轮播图

    一.轮播图 Bootstrap 轮播(Carousel)插件是一种灵活的响应式的向站点添加滑块的方式.除此之外,内容也是足够灵活的,可以是图像.内嵌框架.视频或者其他您想要放置的任何类型的内容. 如果 ...

  3. 浅谈左偏树在OI中的应用

    Preface 可并堆,一个听起来很NB的数据结构,实际上比一般的堆就多了一个合并的操作. 考虑一般的堆合并时,当我们合并时只能暴力把一个堆里的元素一个一个插入另一个堆里,这样复杂度将达到\(\log ...

  4. Luogu P2473 [SCOI2008]奖励关

    比较恶心的概率(期望)+状压DP,想正推2H的我瑟瑟发抖 由于数据范围不大,因此我们可以直接状压每个宝物取或不取的情况,设\(f_{i,j}\)表示前\(i\)轮且宝物是否取过的状态为\(j\)时的方 ...

  5. asp.net core部署时自定义监听端口,提高部署的灵活性

    另一种方式 https://www.cnblogs.com/stulzq/p/9039836.html 代码截图: 贴一下代码,方便复制: //默认端口号5000 string port = &quo ...

  6. Http指南(1)

    网关:是一种特殊的服务器,作为其他服务器的中间实体使用; Agent代理:所有发布web请求的应用程序都是HTTP Agent代理.Web浏览器其实就是一种代理; HTTP报文是在HTTP应用程序之间 ...

  7. Zabbix监控系统部署:配置详解

    1. 全局配置 ListenPort ,监听端口 ,取值范围为1024-32767,默认端口10051 SourceIP,外发连接源地址 LogType,日志类型:单独日志文件,系统文件,控制台输出 ...

  8. 利用阿里云的源yum方式安装Mongodb

    今天在线上服务器上安装MongoDB,从Mongo官网直接下载链接,结果在下载时发觉速度慢的可怜.迫于无奈,只能找国内的镜像下载.这里选择阿里云的源进行安装,记录如下: 1)在/etc/yum.rep ...

  9. Jenkins构建自动化任务

    前言 Jenkins是一个开源软件项目,是基于Java开发的一种持续集成工具,用于监控持续重复的工作,旨在提供一个开放易用的软件平台,使软件的持续集成变成可能. 一.环境配置 1.切换到jenkins ...

  10. CF 910 C. Minimum Sum

    链接 [http://codeforces.com/group/1EzrFFyOc0/contest/910/problem/C] 题意 给你n个字符串,每个字符串的字符是a~j,每个字符都可以是0~ ...