思路:

树链剖分,不知道说什么...我连模板都不会用

代码:

#include<map>
#include<ctime>
#include<cmath>
#include<stack>
#include<queue>
#include<string>
#include<vector>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#define ll long long
const int maxn = 60000+5;
const int INF = 0x3f3f3f3f;
using namespace std;
int n,m,tol;
int cnt;
struct Node{
    int u,v,next;
}node[maxn << 1];
int head[maxn];
int fa[maxn],siz[maxn],dep[maxn],son[maxn],id[maxn],top[maxn];
//父节点,子树大小,深度,重儿子,dfs执行顺序,重链的起始点

void addnode(int u,int v){
    node[tol].u = u;
    node[tol].v = v;
    node[tol].next = head[u];
    head[u] = tol++;
}
void dfs1(int u,int father,int depth){  //确定fa,siz,dep,son
    son[u] = 0;
    dep[u] = depth;
    fa[u] = father;
    siz[u] = 1;
    for(int i = head[u];~i;i = node[i].next){
        int v = node[i].v;
        if(v == father) continue;   //储存无向边,需判断
        dfs1(v,u,depth + 1);
        siz[u] += siz[v];
        if(siz[v] > siz[son[u]])    //更新重儿子
            son[u] = v;
    }
}
void dfs2(int u,int tp){    //按dfs执行顺序编号
    top[u] = tp;    //重链的起始点
    id[u] = ++cnt;  //编号
    if(!son[u]) return;
    //有重儿子,是重链,继续往下
    dfs2(son[u],tp);
    for(int i = head[u];~i;i = node[i].next){
        int v = node[i].v;
        if(v == fa[u] || v == son[u]) continue;
        dfs2(v,v);
    }
}

/*
线段树
*/
int sum[maxn<<2],mx[maxn<<2];

void push_up(int rt){
    sum[rt] = sum[rt << 1] + sum[rt << 1 | 1];
    mx[rt] = max(mx[rt << 1],mx[rt << 1 | 1]);
}
/*void build(int l,int r,int rt){
    sum[rt] = 0;
    mx[rt] = -INF;  //注意
    if(l == r){
        sum[rt] = mx[rt] = val[l];
        return;
    }
    int m = (l + r) >> 1;
    build(l,m,rt << 1);
    build(m + 1,r,rt << 1 | 1);
    push_up(rt);
}*/
void update(int pos,int val,int l,int r,int rt){
    if(l == r){
        sum[rt] = mx[rt] = val;
        return;
    }
    int m = (l + r) >> 1;
    if(pos <= m)
        update(pos,val,l,m,rt << 1);
    else
        update(pos,val,m + 1,r,rt << 1 | 1);
    push_up(rt);
}
int queryS(int L,int R,int l,int r,int rt){
    if(L <= l && R >= r){
        return sum[rt];
    }
    int m = (l + r) >> 1;
    int ans = 0;
    if(L <= m)
        ans += queryS(L,R,l,m,rt << 1);
    if(R > m)
        ans += queryS(L,R,m + 1,r,rt << 1 | 1);
    return ans;
}
int queryM(int L,int R,int l,int r,int rt){
    if(L <= l && R >= r){
        return mx[rt];
    }
    int m = (l + r) >> 1;
    int ans = -INF;
    if(L <= m)
        ans = max(ans,queryM(L,R,l,m,rt << 1));
    if(R > m)
        ans = max(ans,queryM(L,R,m + 1,r,rt << 1 | 1));
    return ans;
}

//////////////////////////////////////////
int FindS(int u,int v){
    int ans = 0;
    while(top[u] != top[v]){    //不在同一条链上,就从u所在转移到v所在
        if(dep[top[u]] < dep[top[v]]){
            swap(u,v);
        }
        ans += queryS(id[top[u]],id[u],1,n,1);
        u = fa[top[u]];
    }
    if(u == v){
        return (ans + queryS(id[u],id[v],1,n,1));
    }
    else{
        if(dep[u] > dep[v]){
            swap(u,v);
        }
        return (ans + queryS(id[u],id[v],1,n,1));
    }
}
int FindM(int u,int v){
    int ans = -INF;
    while(top[u] != top[v]){    //不在同一条链上,就从u所在转移到v所在
        if(dep[top[u]] < dep[top[v]]){
            swap(u,v);
        }
        ans = max(ans,queryM(id[top[u]],id[u],1,n,1));
        u = fa[top[u]];
    }
    if(u == v){
        return max(ans,queryM(id[u],id[v],1,n,1));
    }
    else{
        if(dep[u] > dep[v]){
            swap(u,v);
        }
        return max(ans,queryM(id[u],id[v],1,n,1));
    }
}
///////////////////////////////////////////////////

void init() {
    tol = cnt = 0;
    memset(fa,0,sizeof fa);
    memset(siz,0,sizeof siz);
    memset(son,0,sizeof son);
    memset(dep,0,sizeof dep);
    memset(top,0,sizeof top);
    memset(id,0,sizeof id);
    memset(sum,0,sizeof sum);
    memset(node,0,sizeof node);
    memset(mx,0,sizeof mx);
    memset(head,-1,sizeof head);
}
int main(){
    int q;
    while(~scanf("%d",&n)){
        int u,v;
        init();
        for(int i = 1;i < n;i++){
            scanf("%d%d",&u,&v);
            addnode(u,v);
            addnode(v,u);
        }
        dfs1(1,0,1);
        dfs2(1,1);
        for(int i = 1;i <= n;i++){
            int w;
            scanf("%d",&w);
            update(id[i],w,1,n,1);
        }
        scanf("%d",&q);
        char s[20];
        while(q--){
            scanf("%s%d%d",s,&u,&v);
            if(s[0] == 'C'){
                update(id[u],v,1,n,1);
            }
            else if(s[1] == 'S'){
                printf("%d\n",FindS(u,v));
            }
            else{
                printf("%d\n",FindM(u,v));
            }
        }
    }
    return 0;
}
J

HYSBZ 1036 树的统计Count(树链剖分)题解的更多相关文章

  1. BZOJ 1036: [ZJOI2008]树的统计Count [树链剖分]【学习笔记】

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 14302  Solved: 5779[Submit ...

  2. Bzoj 1036: [ZJOI2008]树的统计Count 树链剖分,LCT

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 11102  Solved: 4490[Submit ...

  3. BZOJ 1036: [ZJOI2008]树的统计Count( 树链剖分 )

    树链剖分... 不知道为什么跑这么慢 = = 调了一节课啊跪.. ------------------------------------------------------------------- ...

  4. bzoj 1036: [ZJOI2008]树的统计Count 树链剖分+线段树

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 16294  Solved: 6645[Submit ...

  5. BZOJ 1036: [ZJOI2008]树的统计Count (树链剖分模板题)

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 14982  Solved: 6081[Submit ...

  6. BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分)(线段树单点修改)

    [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 14968  Solved: 6079[Submit][Stat ...

  7. 【BZOJ1036】[ZJOI2008]树的统计Count 树链剖分

    [BZOJ1036][ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. ...

  8. bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题

    [ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...

  9. Cogs 1688. [ZJOI2008]树的统计Count(树链剖分+线段树||LCT)

    [ZJOI2008]树的统计Count ★★★ 输入文件:bzoj_1036.in 输出文件:bzoj_1036.out 简单对比 时间限制:5 s 内存限制:162 MB [题目描述] 一棵树上有n ...

  10. BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分 - 点权剖分 - 单点权修改)

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1036 树链剖分模版题,打的时候注意点就行.做这题的时候,真的傻了,单词拼错检查了一个多小时 ...

随机推荐

  1. JS-倒计时效果

    团购-限时抢 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www. ...

  2. 【BZOJ3932】[CQOI2015]任务查询系统 主席树

    [BZOJ3932][CQOI2015]任务查询系统 Description 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si, ...

  3. Android 微信分享,分享到朋友圈与分享到好友,以及微信登陆

    extends:http://www.cnblogs.com/android100/p/Android-qq.html 一.申请你的AppID http://open.weixin.qq.com/ 友 ...

  4. centos配置Tomcat以指定的身份(非root)运行

      本文依赖的环境: 已安装并配置好jdk和tomcat环境 已安装并配置好gcc.make等编译工具 1.编译安装守护程序 cd /usr/local/tomcat7/bin/ tar vzxf c ...

  5. 日期提取函数EXTRACT

    EXTRACT extracts and returns the value of a specified datetime field from a datetime or interval exp ...

  6. MDK软件仿真常见问题

    一直不知道MDK该怎么仿真调试程序,之前试了好几次都没有成功.因为有个程序一直不知道里面的变量对应着外部怎么的模式,今天想起可以用仿真调试的方法查看当外部设置某种模式的时候, 内部变量的变化,这样想来 ...

  7. 将web项目打成war包部署在tomcat步骤

    将web项目打成war包部署在tomcat步骤 1.将自己的项目打成war包. 2.将打包好的war复制到${tomcat.home}/webapps项目下. 3.在${tomcat.hom}/con ...

  8. 【gulp】前端自动化工具---gulp的使用(一)------【巷子】

    什么是gulp?   基于node的自动化构建工具   扩展:开发的时候分为2个节点一个是开发阶段  另一个是部署阶段        开发阶段:源文件不会被压缩            部署阶段:所有文 ...

  9. linux漏洞扫描工具【lynis】

    Lynis是一款Unix系统的安全审计以及加固工具,能够进行深层次的安全扫描,其目的是检测潜在的时间并对未来的系统加固提供建议.这款软件会扫描一般系统信息,脆弱软件包以及潜在的错误配置. 特征: 漏洞 ...

  10. poj1821 Fence【队列优化线性DP】

    Fence Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 6122   Accepted: 1972 Description ...